基于Spring AOP @AspectJ进阶说明

@AspectJ可以使用切点函数定义切点,我们还可以使用逻辑运算符对切点进行复核运算得到复合的切点,为了在切面中重用切点,我们还可以对切点进行命名,以便在其他的地方引用定义过的切点。

当一个连接点匹配多个切点时,需要考虑织入顺序的问题,此外一个重要的问题是如何再增强中访问连接点上下文的信息。

Waiter接口:

package com.yyq.aspectJAdvanced;
public interface Waiter {
 void greetTo(String name);
 void serveTo(String name);
}

NaiveWaiter实现类:

package com.yyq.aspectJAdvanced;
public class NaiveWaiter implements Waiter {
 @Override
 public void greetTo(String name) {
  System.out.println("NaiveWaiter:greet to " + name + "...");
 }
 @Override
 public void serveTo(String name) {
  System.out.println("NaiveWaiter:serving to " + name + "...");
 }
 public void smile(String clientName,int times){
  System.out.println("NaiveWaiter:smile to "+clientName+ times+"times...");
 }
}

NaughtyWaiter实现类:

package com.yyq.aspectJAdvanced;
public class NaughtyWaiter implements Waiter {
 public void greetTo(String clientName) {
  System.out.println("NaughtyWaiter:greet to " + clientName + "...");
 }
 public void serveTo(String clientName) {
  System.out.println("NaughtyWaiter:serving " + clientName + "...");
 }
 public void joke(String clientName, int times) {
  System.out.println("NaughtyWaiter:play " + times + " jokes to " + clientName + "...");
 }
}

Seller接口:

package com.yyq.aspectJAdvanced;
public interface Seller {
 int sell(String goods, String clientName);
}

SmallSeller实现类:

package com.yyq.aspectJAdvanced;
public class SmartSeller implements Seller {
 public int sell(String goods,String clientName) {
  System.out.println("SmartSeller: sell "+goods +" to "+clientName+"...");
  return 100;
 }

 public void checkBill(int billId){
  if(billId == 1) throw new IllegalArgumentException("iae Exception");
  else throw new RuntimeException("re Exception");
 }
}

beans.xml配置文件:

<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xmlns:aop="http://www.springframework.org/schema/aop"
  xsi:schemaLocation="http://www.springframework.org/schema/beans
  http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
  http://www.springframework.org/schema/aop
  http://www.springframework.org/schema/aop/spring-aop-3.0.xsd">
 <aop:aspectj-autoproxy proxy-target-class="true"/>
 <bean id="naiveWaiter" class="com.yyq.aspectJAdvanced.NaiveWaiter"/>
 <bean id="naughtyWaiter" class="com.yyq.aspectJAdvanced.NaughtyWaiter"/>
 <bean id="seller" class="com.yyq.aspectJAdvanced.SmartSeller"/>
 <!--
 <bean class="com.yyq.aspectJAdvanced.TestAspect"/>

 <bean class="com.yyq.aspectJAdvanced.TestAspect2"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect3"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect4"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect5"/>
 <bean id="naiveWaiter2" class="com.yyq.aspectJAdvanced.NaiveWaiter2"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect6"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect7"/>
 <bean class="com.yyq.aspectJAdvanced.TestAspect8"/>
-->
</beans>

1、切点符合运算

使用切点符合运算符,我们将拥有强大而灵活的切点表达能力。

TestAspect:切点符合运算定义切面

package com.yyq.aspectJAdvanced;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class TestAspect {
 //与非运算
 @Before("!target(com.yyq.aspectJAdvanced.NaiveWaiter) && execution(* serveTo(..))")
 public void notServeInNaiveWaiter(){
  System.out.println("--notServeInNaiveWaiter() executed!--");
 }
 //与运算
 @After("within(com.yyq.aspectJAdvanced.*) && execution(* greetTo(..))")
 public void greetToFun(){
  System.out.println("--greetToFun() executed!--");
 }
 //或运算
 @AfterReturning("target(com.yyq.aspectJAdvanced.Waiter) || target(com.yyq.aspectJAdvanced.Seller)")
 public void waiterOrSeller(){
  System.out.println("--waiterOrSeller() executed!--");
 }
}

测试方法:

@Test
 public void pointAspectJTest() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter naiveWaiter = (Waiter) ctx.getBean("naiveWaiter");
  Waiter naughtyWaiter = (Waiter) ctx.getBean("naughtyWaiter");
  naiveWaiter.greetTo("John");
  naiveWaiter.serveTo("John");
  naughtyWaiter.greetTo("Tom");
  naughtyWaiter.serveTo("Tom");
 }

输出结果:

NaiveWaiter:greet to John...
--greetToFun() executed!--
--waiterOrSeller() executed!--
NaiveWaiter:serving to John...
--waiterOrSeller() executed!--
NaughtyWaiter:greet to Tom...
--greetToFun() executed!--
--waiterOrSeller() executed!--
--notServeInNaiveWaiter() executed!--
NaughtyWaiter:serving Tom...
--waiterOrSeller() executed!--

2、命名切点

切点直接声明在增强方法处被称为匿名切点,匿名切点只能在声明处使用。如果希望在其他地方重用一个切点,我们可以通过@Pointcut注解以及切面类方法对切点进行命名。

TestNamePointcut:命名切点类

package com.yyq.aspectJAdvanced;
import org.aspectj.lang.annotation.Pointcut;
public class TestNamePointcut {
 //通过注解方法inPackage()对该切点进行命名,方法可视域修饰符为private,表明该命名切点只能在本切面类中使用
 @Pointcut("within(com.yyq.aspectJAdvaned.*)")
 private void inPackage(){}
 @Pointcut("execution(* greetTo(..))")
 protected void greetTo(){}
 @Pointcut("inPackage() and greetTo()")
 public void inPkgGreetTo(){}
}

TestAspect2:切面实现类

package com.yyq.aspectJAdvanced;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
@Aspect
public class TestAspect2 {
 @Before("TestNamePointcut.inPkgGreetTo()")
 public void pkgGreetTo(){
  System.out.println("--pkgGreetTo() executed!--");
 }
 @Before("target(com.yyq.aspectJAdvanced.NaiveWaiter) || TestNamePointcut.inPkgGreetTo()")
 public void pkgGreetToNotnaiveWaiter(){
  System.out.println("--pkgGreetToNotnaiveWaiter() executed!--");
 }
}

测试方法:

@Test
 public void pointAspectJTest2() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  NaiveWaiter naiveWaiter = (NaiveWaiter) ctx.getBean("naiveWaiter");
  naiveWaiter.smile("Andy", 2);
 }

输出结果:

--pkgGreetToNotnaiveWaiter() executed!--
NaiveWaiter:smile to Andy2times...

3、增强织入的顺序

一个连接点可以同时匹配多个切点,切点对应的增强在连接点上的织入顺序的安排主要有以下3种情况:

1)如果增强在同一个切面类中声明,则依照增强在切面类中定义的顺序进行织入;

2)如何增强位于不同的切面类中,且这些切面类都实现了org.springframework.core.Order接口,则由接口方法的顺序号决定(顺序号小的先织入);

3)如果增强位于不同的切面类中,且这些切面类没有实现org.springframework.core.Order接口,织入的顺序是不确定的。

4、访问连接点信息

AspectJ使用org.aspectj.lang.JoinPoint接口表示目标类连接点对象,如果是环绕增强时,使用org.aspectj.lang.ProceedingJoinPoint表示连接点对象,该类是JoinPoint的子接口,任何一个增强方法都可以通过将第一个入参声明为JoinPoint访问到连接点上下文的信息。

TestAspect3:切面实现类

@Aspect
public class TestAspect3 {
 @Around("execution(* greetTo(..)) && target(com.yyq.aspectJAdvanced.NaiveWaiter)")
 public void joinPointAccess(ProceedingJoinPoint pjp) throws Throwable {
  System.out.println("---joinPointAccess---");
  System.out.println("args[0]:" + pjp.getArgs()[0]);
  System.out.println("signature:" + pjp.getTarget().getClass());
  pjp.proceed();
  System.out.println("---joinPointAccess---");
 }
}

测试方法:

 @Test
 public void pointAspectJTest3() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter naiveWaiter = (Waiter) ctx.getBean("naiveWaiter");
  naiveWaiter.greetTo("Andy");
 }

输出结果:

---joinPointAccess---
args[0]:Andy
signature:class com.yyq.aspectJAdvanced.NaiveWaiter
NaiveWaiter:greet to Andy...
---joinPointAccess---

5、绑定连接点方法入参

args()用于绑定连接点方法的入参;@annotation()用于绑定连接点方法的注解对象;而@args()用于绑定连接点方法入参的注解。

TestAspect4:切面实现类

@Aspect
public class TestAspect4 {
 @Before("target(com.yyq.aspectJAdvanced.NaiveWaiter) && args(name,num,..)")
 public void bindJoinPointParams(int num, String name) {
  System.out.println("---bindJoinPointParams---");
  System.out.println("name:" + name);
  System.out.println("num:" + num);
  System.out.println("---bindJoinPointParams---");
 }
}

测试方法:

@Test
 public void pointAspectJTest4() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  NaiveWaiter naiveWaiter = (NaiveWaiter) ctx.getBean("naiveWaiter");
  naiveWaiter.smile("Andy", 3);
 }

输出结果:

---bindJoinPointParams---
name:Andy
num:3
---bindJoinPointParams---
NaiveWaiter:smile to Andy 3 times...

6、绑定代理对象

使用this()或target()可绑定被代理对象实例,在通过类实例名绑定对象时,还依然具有原来连接点匹配的功能,只不过类名是通过增强方法中同名入参的类型间接决定罢了。

TestAspect5:切面实现类

@Aspect
public class TestAspect5 {
 @Before("this(waiter)")
 public void bindProxyObj(Waiter waiter){
  System.out.println("---bindProxyObj---");
  System.out.println(waiter.getClass().getName());
  System.out.println("---bindProxyObj---");
 }
}

测试方法:

@Test
 public void pointAspectJTest5() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter waiter = (Waiter) ctx.getBean("naiveWaiter");
  waiter.greetTo("Yang");
 }

输出结果:

---bindProxyObj---
com.yyq.aspectJAdvanced.NaiveWaiter$$EnhancerByCGLIB$$fefafe52
---bindProxyObj---
NaiveWaiter:greet to Yang...

7、绑定类注解对象

@within()和@target()函数可以将目标类的注解对象绑定到增强方法中,我们通过@within()演示注解绑定的操作。

TestAspect6:切面测试类

@Aspect
public class TestAspect6 {
 @Before("@within(m)")
 public void bindTypeAnnoObject(Monitorable m) {
  System.out.println("---bindTypeAnnoObject---");
  System.out.println(m.getClass().getName());
  System.out.println("---bindTypeAnnoObject---");
 }
}

测试方法:

@Test
 public void pointAspectJTest6() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  Waiter waiter = (Waiter) ctx.getBean("naiveWaiter2");
  ((NaiveWaiter2)waiter).greetTo("Yang");
 }

输出结果:

---bindTypeAnnoObject---
$Proxy4
---bindTypeAnnoObject---
NaiveWaiter:greet to Yang...

8、绑定返回值

在后置增强中,我们可以通过returning绑定连接点方法的返回值。

TestAspect7:切面实现类

@Aspect
public class TestAspect7 {
 @AfterReturning(value = "target(com.yyq.aspectJAdvanced.SmartSeller)", returning = "retVal")
 public void bindReturnValue(int retVal) {
  System.out.println("---bindReturnValue---");
  System.out.println("returnValue:" + retVal);
  System.out.println("---bindReturnValue---");
 }
}

测试方法:

 @Test
 public void pointAspectJTest7() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  SmartSeller seller = (SmartSeller) ctx.getBean("seller");
  seller.sell("Beer", "John");
 } 

输出结果:

SmartSeller: sell Beer to John...
---bindReturnValue---
returnValue:100
---bindReturnValue---

9、绑定抛出的异常

和通过切点函数绑定连接点信息不同,连接点抛出的异常必须使用AfterThrowing注解的throwing成员进行绑定。

TestAspect8:切面实现类

@Aspect
public class TestAspect8 {
 @AfterThrowing(value = "target(com.yyq.aspectJAdvanced.SmartSeller)", throwing = "iae")
 public void bindException(IllegalArgumentException iae) {
  System.out.println("---bindException---");
  System.out.println("exception:" + iae.getMessage());
  System.out.println("---bindException---");
 }
}

测试方法:

 @Test
 public void pointAspectJTest8() {
  String configPath = "com\\yyq\\aspectJAdvanced\\beans.xml";
  ApplicationContext ctx = new ClassPathXmlApplicationContext(configPath);
  SmartSeller seller = (SmartSeller) ctx.getBean("seller");
  seller.checkBill(1);
 }

输出结果:

---bindException---
exception:iae Exception
---bindException---

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。如有错误或未考虑完全的地方,望不吝赐教。

(0)

相关推荐

  • Spring AOP的使用详解

    什么是AOP AOP(Aspect Oriented Programming 面向切面编程),通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型.利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率. 常用于日志记录,性能统计,安全控制,事务处理,异常处理等等. 定义AOP术语 切面(Aspect):切

  • 谈谈Spring AOP中@Aspect的高级用法示例

    前言 本文主要跟大家分享介绍了关于Spring AOP中@Aspect的高级用法,下面话不多说了,来随着小编一起看看详细的介绍吧. 1 切点复合运算 支持在切点定义中加入以下运算符进行复合运算: 运算符 说明 && 与运算. ! 非运算. || 或运算. 2 切点命名 一般情况下,切点是直接声明在需要增强方法处,这种切点的声明方式称为匿名切点,匿名切点只能在声明处被使用 . 如果希望在其它地方可以重用这个切点,我们可以通过 @Pointcut 注解及切面类方法来命名它. public cl

  • Spring AspectJ AOP框架注解原理解析

    什么是AspectJ AspectJ是一个面向切面的框架,它扩展了Java语言.AspectJ定义了AOP语法所以它有一个专门的编译器用来生成遵守Java字节编码规范的Class文件. AspectJ是一个基于Java语言的AOP框架 Spring2.0以后新增了对AspectJ切点表达式支持 @AspectJ 是AspectJ1.5新增功能,通过JDK5注解技术,允许直接在Bean类中定义切面 新版本Spring框架,建议使用AspectJ方式来开发AOP AspectJ表达式: 语法:exe

  • 基于Spring AOP @AspectJ进阶说明

    @AspectJ可以使用切点函数定义切点,我们还可以使用逻辑运算符对切点进行复核运算得到复合的切点,为了在切面中重用切点,我们还可以对切点进行命名,以便在其他的地方引用定义过的切点. 当一个连接点匹配多个切点时,需要考虑织入顺序的问题,此外一个重要的问题是如何再增强中访问连接点上下文的信息. Waiter接口: package com.yyq.aspectJAdvanced; public interface Waiter { void greetTo(String name); void se

  • Spring AOP AspectJ使用及配置过程解析

    这篇文章主要介绍了Spring AOP AspectJ使用及配置过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 AspectJ是一个基于Java语言的AOP框架,Spring2.0以后新增了对AspectJ切点表达式支持.因为Spring1.0的时候Aspectj还未出现; AspectJ1.5中新增了对注解的支持,允许直接在Bean类中定义切面.新版本的Spring框架建 议我们都使用AspectJ方式来开发AOP,并提供了非常灵活且

  • 基于 Spring Aop 环绕通知实现 Redis 缓存双删功能(示例代码)

    基于 spring aop 常规应用场景多是用于日志记录以及实现 redis 分布式锁,在 github 中也有项目是把它拿来当作缓存的异常捕捉.从而避免影响实际业务的开发:在某天,笔者有个业务开发是给某个服务模块增加 redis 缓存.增加缓存就会涉及 redis 删除.所以笔者就在思考是不是可以用环绕通知的方式来进行实现 代码实现 结构示意图: 自定义注解 RedisDelByDbUpdate @Repeatable 表示允许在同一个地方上使用相同的注解,没有该注解时贴相同注解会报错 @Ta

  • 基于spring AOP @Around @Before @After的区别说明

    此段小代码演示了spring aop中@Around @Before @After三个注解的区别 @Before是在所拦截方法执行之前执行一段逻辑. @After 是在所拦截方法执行之后执行一段逻辑. @Around是可以同时在所拦截方法的前后执行一段逻辑. 连接点(JoinPoint) 这个就更好解释了,就是spring允许你是通知(Advice)的地方,那可就真多了,基本每个方法的前.后(两者都有也行),或抛出异常是时都可以是连接点,spring只支持方法连接点. 其他如AspectJ还可以

  • 基于Spring AOP proxyTargetClass的行为表现总结

    Spring AOP proxyTargetClass的行为 要点列表形式 proxyTargetClass true 目标对象实现了接口 – 使用CGLIB代理机制 目标对象没有接口(只有实现类) – 使用CGLIB代理机制 false 目标对象实现了接口 – 使用JDK动态代理机制(代理所有实现了的接口) 目标对象没有接口(只有实现类) – 使用CGLIB代理机制 表格形式 proxyTargetClass 目标对象特征 代理效果 true 目标对象实现了接口 使用CGLIB代理机制 tru

  • Aspectj与Spring AOP的对比分析

    1.简介 今天有多个可用的 AOP 库, 它们需要能够回答许多问题: 1.是否与用户现有的或新的应用程序兼容? 2.在哪里可以实现 AOP? 3.与自己的应用程序集成多快? 4.性能开销是多少? 在本文中, 我们将研究如何回答这些问题, 并介绍 Spring aop 和 AspectJ, 这是 Java 的两个最受欢迎的 aop 框架. 2.AOP概念 在开始之前, 让我们对术语和核心概念进行快速.高层次的审查: Aspect -- 一种标准代码/功能, 分散在应用程序中的多个位置, 通常与实际

  • Spring AOP 自定义注解的实现代码

    1.在Maven中加入以下以依赖: <!-- Spring AOP + AspectJ by shipengzhi --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>3.0.6.RELEASE</version> </dependency> <

  • Spring AOP切面解决数据库读写分离实例详解

    Spring AOP切面解决数据库读写分离实例详解 为了减轻数据库的压力,一般会使用数据库主从(master/slave)的方式,但是这种方式会给应用程序带来一定的麻烦,比如说,应用程序如何做到把数据写到master库,而读取数据的时候,从slave库读取.如果应用程序判断失误,把数据写入到slave库,会给系统造成致命的打击. 解决读写分离的方案很多,常用的有SQL解析.动态设置数据源.SQL解析主要是通过分析sql语句是insert/select/update/delete中的哪一种,从而对

  • java基于spring注解AOP的异常处理的方法

    一.前言 项目刚刚开发的时候,并没有做好充足的准备.开发到一定程度的时候才会想到还有一些问题没有解决.就比如今天我要说的一个问题:异常的处理.写程序的时候一般都会通过try...catch...finally对异常进行处理,但是我们真的能在写程序的时候处理掉所有可能发生的异常吗? 以及发生异常的时候执行什么逻辑,返回什么提示信息,跳转到什么页面,这些都是要考虑到的. 二.基于@ControllerAdvice(加强的控制器)的异常处理 @ControllerAdvice注解内部使用@Except

  • Spring AOP 基于注解详解及实例代码

    Spring AOP  基于注解详解及实例代码 1.启用spring对@AspectJ注解的支持: <beans xmlns:aop="http://www.springframework.org/schema/aop"...> <!--启动支持--> <aop:aspectj-autoproxy /> </beans> 也可以配置AnnotationAwareAspectJAutoProxyCreator Bean来启动Spring对@

随机推荐