34.@Aspect中@Pointcut 12种用法

本文继续AOP,目前手动Aop中三种方式已经介绍2种了,本文将介绍另外一种:AspectJProxyFactory,可能大家对这个比较陌生,但是@Aspect这个注解大家应该很熟悉吧,通过这个注解在spring环境中实现aop特别的方便。

AspectJProxyFactory这个类可以通过解析@Aspect标注的类来生成代理aop代理对象,对开发者来说,使创建代理变的更简洁了。

先了解几个概念

文中会涉及几个概念,先了解一下。

target

用来表示目标对象,即需要通过aop来增强的对象。

proxy

代理对象,target通过aop增强之后生成的代理对象。

AspectJ

AspectJ是什么?

AspectJ是一个面向切面的框架,是目前最好用,最方便的AOP框架,和spring中的aop可以集成在一起使用,通过Aspectj提供的一些功能实现aop代理变得非常方便。

AspectJ使用步骤

 
  1. 1.创建一个类,使用@Aspect标注
  2. 2.@Aspect标注的类中,通过@Pointcut定义切入点
  3. 3.@Aspect标注的类中,通过AspectJ提供的一些通知相关的注解定义通知
  4. 4.使用AspectJProxyFactory结合@Ascpect标注的类,来生成代理对象

先来个案例,感受一下AspectJ是多么的方便。

来个类

 
  1. package com.javacode2018.aop.demo9.test1;
  2. public class Service1 {
  3. public void m1() {
  4. System.out.println("我是 m1 方法");
  5. }
  6. public void m2() {
  7. System.out.println(10 / 0);
  8. System.out.println("我是 m2 方法");
  9. }
  10. }

通过AspectJ来对Service1进行增强,来2个通知,一个前置通知,一个异常通知,这2个通知需要对Service1中的所有方法生效,实现如下:

 
  1. package com.javacode2018.aop.demo9.test1;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.AfterThrowing;
  4. import org.aspectj.lang.annotation.Aspect;
  5. import org.aspectj.lang.annotation.Before;
  6. import org.aspectj.lang.annotation.Pointcut;
  7. //@1:这个类需要使用@Aspect进行标注
  8. @Aspect
  9. public class Aspect1 {
  10. //@2:定义了一个切入点,可以匹配Service1中所有方法
  11. @Pointcut("execution(* com.javacode2018.aop.demo9.test1.Service1.*(..))")
  12. public void pointcut1() {
  13. }
  14. //@3:定义了一个前置通知,这个通知对刚刚上面我们定义的切入点中的所有方法有效
  15. @Before(value = "pointcut1()")
  16. public void before(JoinPoint joinPoint) {
  17. //输出连接点的信息
  18. System.out.println("前置通知," + joinPoint);
  19. }
  20. //@4:定义了一个异常通知,这个通知对刚刚上面我们定义的切入点中的所有方法有效
  21. @AfterThrowing(value = "pointcut1()", throwing = "e")
  22. public void afterThrowing(JoinPoint joinPoint, Exception e) {
  23. //发生异常之后输出异常信息
  24. System.out.println(joinPoint + ",发生异常:" + e.getMessage());
  25. }
  26. }

@1:类上使用@Aspect标注

@2:通过@Pointcut注解标注在方法上面,用来定义切入点

@3:使用@Before标注在方法上面,定义了一个前置通知,通过value引用了上面已经定义的切入点,表示这个通知会对Service1中的所有方法生效,在通知中可以通过这个类名.方法名()引用@Pointcut定义的切入点,表示这个通知对这些切入点有效,若@Before和@Pointcut在一个类的时候,直接通过方法名()引用当前类中定义的切入点

@4:这个使用@AfterThrowing定义了一个异常通知,也是对通过value引用了上面已经定义的切入点,表示这个通知会对Service1中的所有方法生效,若Service1中的方法抛出了Exception类型的异常,都会回调afterThrowing方法。

来个测试类

 
  1. package com.javacode2018.aop.demo9;
  2. import com.javacode2018.aop.demo9.test1.Aspect1;
  3. import com.javacode2018.aop.demo9.test1.Service1;
  4. import org.junit.Test;
  5. import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;
  6. public class AopTest9 {
  7. @Test
  8. public void test1() {
  9. try {
  10. //对应目标对象
  11. Service1 target = new Service1();
  12. //创建AspectJProxyFactory对象
  13. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  14. //设置被代理的目标对象
  15. proxyFactory.setTarget(target);
  16. //设置标注了@Aspect注解的类
  17. proxyFactory.addAspect(Aspect1.class);
  18. //生成代理对象
  19. Service1 proxy = proxyFactory.getProxy();
  20. //使用代理对象
  21. proxy.m1();
  22. proxy.m2();
  23. } catch (Exception e) {
  24. }
  25. }
  26. }

运行输出

 
  1. 前置通知,execution(void com.javacode2018.aop.demo9.test1.Service1.m1())
  2. 我是 m1 方法
  3. 前置通知,execution(void com.javacode2018.aop.demo9.test1.Service1.m2())
  4. execution(void com.javacode2018.aop.demo9.test1.Service1.m2()),发生异常:/ by zero

使用是不是特方便。

AspectJProxyFactory原理

@Aspect标注的类上,这个类中,可以通过通过@Pointcut来定义切入点,可以通过@Before、@Around、@After、@AfterRunning、@AfterThrowing标注在方法上来定义通知,定义好了之后,将@Aspect标注的这个类交给AspectJProxyFactory来解析生成Advisor链,进而结合目标对象一起来生成代理对象,大家可以去看一下源码,比较简单,这里就不多解释了。

本文的重点在@Aspect标注的类上,@Aspect中有2个关键点比较重要

  • @Pointcut:标注在方法上,用来定义切入点,有11种用法,本文主要讲解这11种用法。
  • @Aspect类中定义通知:可以通过@Before、@Around、@After、@AfterRunning、@AfterThrowing标注在方法上来定义通知,这个下一篇介绍。

@Pointcut的12种用法

作用

用来标注在方法上来定义切入点。

定义

格式:@ 注解(value=“表达标签 (表达式格式)”)

如:

 
  1. @Pointcut("execution(* com.javacode2018.aop.demo9.test1.Service1.*(..))")

表达式标签(10种)

  • execution:用于匹配方法执行的连接点
  • within:用于匹配指定类型内的方法执行
  • this:用于匹配当前AOP代理对象类型的执行方法;注意是AOP代理对象的类型匹配,这样就可能包括引入接口的类型匹配
  • target:用于匹配当前目标对象类型的执行方法;注意是目标对象的类型匹配,这样就不包括引入接口的类型匹配
  • args:用于匹配当前执行的方法传入的参数为指定类型的执行方法
  • @within:用于匹配所以持有指定注解类型内的方法
  • @target:用于匹配当前目标对象类型的执行方法,其中目标对象持有指定的注解
  • @args:用于匹配当前执行的方法传入的参数持有指定注解的执行
  • @annotation:用于匹配当前执行方法持有指定注解的方法
  • bean:Spring AOP扩展的,AspectJ没有对于指示符,用于匹配特定名称的Bean对象的执行方法

10种标签组成了12种用法

1、execution

使用execution(方法表达式)匹配方法执行。

execution格式
 
  1. execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)
  • 其中带 ?号的 modifiers-pattern?,declaring-type-pattern?,hrows-pattern?是可选项
  • ret-type-pattern,name-pattern, parameters-pattern是必选项
  • modifier-pattern? 修饰符匹配,如public 表示匹配公有方法
  • ret-type-pattern 返回值匹配,* 表示任何返回值,全路径的类名等
  • declaring-type-pattern? 类路径匹配
  • name-pattern 方法名匹配, 代表所有,set,代表以set开头的所有方法
  • (param-pattern) 参数匹配,指定方法参数(声明的类型),(..)代表所有参数,(*,String)代表第一个参数为任何值,第二个为String类型,(..,String)代表最后一个参数是String类型
  • throws-pattern? 异常类型匹配
举例说明
表达式描述
public *.*(..)任何公共方法的执行
* com.javacode2018..IPointcutService.*()com.javacode2018包及所有子包下IPointcutService接口中的任何无参方法
* com.javacode2018..*.*(..)com.javacode2018包及所有子包下任何类的任何方法
* com.javacode2018..IPointcutService.*(*)com.javacode2018包及所有子包下IPointcutService接口的任何只有一个参数方法
* com.javacode2018..IPointcutService+.*()com.javacode2018包及所有子包下IPointcutService接口及子类型的的任何无参方法
* Service1.*(String)匹配Service1中只有1个参数的且参数类型是String的方法
* Service1.*(*,String)匹配Service1中只有2个参数的且第二个参数类型是String的方法
* Service1.*(..,String)匹配Service1中最后1个参数类型是String的方法
类型匹配语法

很多地方会按照类型的匹配,先来说一下类型匹配的语法。

首先让我们来了解下AspectJ类型匹配的通配符:

  • *:匹配任何数量字符
  • ..:匹配任何数量字符的重复,如在类型模式中匹配任何数量子包;而在方法参数模式中匹配任何数量参数(0个或者多个参数)
  • +:匹配指定类型及其子类型;仅能作为后缀放在类型模式后边
表达式说明
java.lang.String匹配String类型
java.*.String匹配java包下的任何一级子包下的String类型,如匹配java.lang.String,但不匹配java.lang.ss.String
java..*匹配java包及任何子包下的任何类型,如匹配java.lang.String、java.lang.annotation.Annotation
java.lang.*ing匹配任何java.lang包下的以ing结尾的类型
java.lang.Number+匹配java.lang包下的任何Number类型及其子类型,如匹配java.lang.Number,也匹配java.lang.Integer、java.math.BigInteger

2、within

用法

within(类型表达式):目标对象target的类型是否和within中指定的类型匹配

表达式描述
within(com.javacode2018..*)com.javacode2018包及子包下的任何方法执行
within(com.javacode2018..IPointcutService+)com.javacode2018包或所有子包下IPointcutService类型及子类型的任何方法
within(com.javacode2018.Service1)匹配类com.javacode2018.Service1中定义的所有方法,不包含其子类中的方法
匹配原则
 
  1. target.getClass().equals(within表达式中指定的类型)
案例

有2个类,父子关系

父类C1

 
  1. package com.javacode2018.aop.demo9.test2;
  2. public class C1 {
  3. public void m1() {
  4. System.out.println("我是m1");
  5. }
  6. public void m2() {
  7. System.out.println("我是m2");
  8. }
  9. }

子类C2

 
  1. package com.javacode2018.aop.demo9.test2;
  2. public class C2 extends C1 {
  3. @Override
  4. public void m2() {
  5. super.m2();
  6. }
  7. public void m3() {
  8. System.out.println("我是m3");
  9. }
  10. }

来个Aspect类

 
  1. package com.javacode2018.aop.demo9.test2;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest2 {
  8. @Pointcut("within(C1)") //@1
  9. public void pc() {
  10. }
  11. @Before("pc()") //@2
  12. public void beforeAdvice(JoinPoint joinpoint) {
  13. System.out.println(joinpoint);
  14. }
  15. }

注意@1匹配的类型是C1,也就是说被代理的对象的类型必须是C1类型的才行,需要和C1完全匹配

下面我们对C2创建代理

 
  1. @Test
  2. public void test2(){
  3. C2 target = new C2();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest2.class);
  7. C2 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. proxy.m2();
  10. proxy.m3();
  11. }

运行输出

 
  1. 我是m1
  2. 我是m2
  3. 我是m3

原因是目标对象是C2类型的,C2虽然是C1的子类,但是within中表达式指定的是要求类型必须是C1类型的才匹配。

如果将within表达式修改为下面任意一种就可以匹配了

 
  1. @Pointcut("within(C1+)")
  2. @Pointcut("within(C2)")

再次运行输出

 
  1. execution(void com.javacode2018.aop.demo9.test2.C1.m1())
  2. 我是m1
  3. execution(void com.javacode2018.aop.demo9.test2.C2.m2())
  4. 我是m2
  5. execution(void com.javacode2018.aop.demo9.test2.C2.m3())
  6. 我是m3

3、this

用法

this(类型全限定名):通过aop创建的代理对象的类型是否和this中指定的类型匹配;注意判断的目标是代理对象;this中使用的表达式必须是类型全限定名,不支持通配符。

匹配原则
 
  1. 如:this(x),则代理对象proxy满足下面条件时会匹配
  2. x.getClass().isAssignableFrom(proxy.getClass());
案例

来个接口

 
  1. package com.javacode2018.aop.demo9.test3;
  2. public interface I1 {
  3. void m1();
  4. }

来个实现类

 
  1. package com.javacode2018.aop.demo9.test3;
  2. public class Service3 implements I1 {
  3. @Override
  4. public void m1() {
  5. System.out.println("我是m1");
  6. }
  7. }

来个@Aspect类

 
  1. package com.javacode2018.aop.demo9.test3;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest3 {
  8. //@1:匹配proxy是Service3类型的所有方法
  9. @Pointcut("this(Service3)")
  10. public void pc() {
  11. }
  12. @Before("pc()")
  13. public void beforeAdvice(JoinPoint joinpoint) {
  14. System.out.println(joinpoint);
  15. }
  16. }

测试代码

 
  1. @Test
  2. public void test3() {
  3. Service3 target = new Service3();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. //获取目标对象上的接口列表
  7. Class<?>[] allInterfaces = ClassUtils.getAllInterfaces(target);
  8. //设置需要代理的接口
  9. proxyFactory.setInterfaces(allInterfaces);
  10. proxyFactory.addAspect(AspectTest3.class);
  11. //获取代理对象
  12. Object proxy = proxyFactory.getProxy();
  13. //调用代理对象的方法
  14. ((I1) proxy).m1();
  15. System.out.println("proxy是否是jdk动态代理对象:" + AopUtils.isJdkDynamicProxy(proxy));
  16. System.out.println("proxy是否是cglib代理对象:" + AopUtils.isCglibProxy(proxy));
  17. //判断代理对象是否是Service3类型的
  18. System.out.println(Service3.class.isAssignableFrom(proxy.getClass()));
  19. }

运行输出

 
  1. 我是m1
  2. proxy是否是jdk动态代理对象:true
  3. proxy是否是cglib代理对象:false
  4. false

从输出中可以看出m1方法没有被增强,原因:this表达式要求代理对象必须是Service3类型的,输出中可以看出代理对象并不是Service3类型的,此处代理对象proxy是使用jdk动态代理生成的。

我们可以将代码调整一下,使用cglib来创建代理

 
  1. proxyFactory.setProxyTargetClass(true);

再次运行,会发现m2被拦截了,结果如下

 
  1. execution(void com.javacode2018.aop.demo9.test3.Service3.m1())
  2. 我是m1
  3. proxy是否是jdk动态代理对象:false
  4. proxy是否是cglib代理对象:true
  5. true

4、target

用法

target(类型全限定名):判断目标对象的类型是否和指定的类型匹配;注意判断的是目标对象的类型;表达式必须是类型全限定名,不支持通配符。

匹配原则
 
  1. 如:target(x),则目标对象target满足下面条件时会匹配
  2. x.getClass().isAssignableFrom(target.getClass());
案例
 
  1. package com.javacode2018.aop.demo9.test4;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest4 {
  8. //@1:目标类型必须是Service3类型的
  9. @Pointcut("target(com.javacode2018.aop.demo9.test3.Service3)")
  10. public void pc() {
  11. }
  12. @Before("pc()")
  13. public void beforeAdvice(JoinPoint joinpoint) {
  14. System.out.println(joinpoint);
  15. }
  16. }

测试代码

 
  1. @Test
  2. public void test4() {
  3. Service3 target = new Service3();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setProxyTargetClass(true);
  6. proxyFactory.setTarget(target);
  7. proxyFactory.addAspect(AspectTest4.class);
  8. //获取代理对象
  9. Object proxy = proxyFactory.getProxy();
  10. //调用代理对象的方法
  11. ((I1) proxy).m1();
  12. //判断target对象是否是Service3类型的
  13. System.out.println(Service3.class.isAssignableFrom(target.getClass()));
  14. }

运行输出

 
  1. execution(void com.javacode2018.aop.demo9.test3.Service3.m1())
  2. 我是m1
  3. true
within、this、target对比
表达式标签判断的对象判断规则(x:指表达式中指定的类型)
withintarget对象target.getClass().equals(表达式中指定的类型)
thisproxy对象x.getClass().isAssignableFrom(proxy.getClass());
targettarget对象x.getClass().isAssignableFrom(target.getClass());

5、args

用法

args(参数类型列表)匹配当前执行的方法传入的参数是否为args中指定的类型;注意是匹配传入的参数类型,不是匹配方法签名的参数类型;参数类型列表中的参数必须是类型全限定名,不支持通配符;args属于动态切入点,也就是执行方法的时候进行判断的,这种切入点开销非常大,非特殊情况最好不要使用。

举例说明
表达式描述
args(String)匹配只有一个参数且传入的参数类型是String类型的方法
args(*,String)匹配只有2个参数的且第2个参数类型是String的方法
args(..,String)匹配最后1个参数类型是String的方法
案例

下面的m1方法参数是Object类型的。

 
  1. package com.javacode2018.aop.demo9.test5;
  2. public class Service5 {
  3. public void m1(Object object) {
  4. System.out.println("我是m1方法,参数:" + object);
  5. }
  6. }

Aspect类

 
  1. package com.javacode2018.aop.demo9.test5;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. import java.util.Arrays;
  7. import java.util.stream.Collectors;
  8. @Aspect
  9. public class AspectTest5 {
  10. //@1:匹配只有1个参数其类型是String类型的
  11. @Pointcut("args(String)")
  12. public void pc() {
  13. }
  14. @Before("pc()")
  15. public void beforeAdvice(JoinPoint joinpoint) {
  16. System.out.println("请求参数:" + Arrays.stream(joinpoint.getArgs()).collect(Collectors.toList()));
  17. }
  18. }

测试代码,调用2次m1方法,第一次传入一个String类型的,第二次传入一个int类型的,看看效果

 
  1. @Test
  2. public void test5() {
  3. Service5 target = new Service5();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest5.class);
  7. Service5 proxy = proxyFactory.getProxy();
  8. proxy.m1("路人");
  9. proxy.m1(100);
  10. }

运行输出

 
  1. 请求参数:[路人]
  2. 我是m1方法,参数:路人
  3. 我是m1方法,参数:100

输出中可以看出,m1第一次调用被增强了,第二次没有被增强。

args会在调用的过程中对参数实际的类型进行匹配,比较耗时,慎用。

6、@within

用法

@within(注解类型):匹配指定的注解内定义的方法。

匹配规则

调用目标方法的时候,通过java中Method.getDeclaringClass()获取当前的方法是哪个类中定义的,然后会看这个类上是否有指定的注解。

 
  1. 被调用的目标方法Method对象.getDeclaringClass().getAnnotation(within中指定的注解类型) != null

来看3个案例。

案例1

目标对象上有@within中指定的注解,这种情况时,目标对象的所有方法都会被拦截。

来个注解
 
  1. package com.javacode2018.aop.demo9.test9;
  2. import java.lang.annotation.ElementType;
  3. import java.lang.annotation.Retention;
  4. import java.lang.annotation.RetentionPolicy;
  5. import java.lang.annotation.Target;
  6. @Retention(RetentionPolicy.RUNTIME)
  7. @Target(ElementType.TYPE)
  8. public @interface Ann9 {
  9. }
来个目标类,用@Ann9标注
 
  1. package com.javacode2018.aop.demo9.test9;
  2. @Ann9
  3. public class S9 {
  4. public void m1() {
  5. System.out.println("我是m1方法");
  6. }
  7. }
来个Aspect类
 
  1. package com.javacode2018.aop.demo9.test9;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest9 {
  8. /**
  9. * 定义目标方法的类上有Ann9注解
  10. */
  11. @Pointcut("@within(Ann9)")
  12. public void pc() {
  13. }
  14. @Before("pc()")
  15. public void beforeAdvice(JoinPoint joinPoint) {
  16. System.out.println(joinPoint);
  17. }
  18. }
测试代码
 
  1. @Test
  2. public void test9() {
  3. S9 target = new S9();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest9.class);
  7. S9 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. }

m1方法在类S9中定义的,S9上面有Ann9注解,所以匹配成功

运行输出
 
  1. execution(void com.javacode2018.aop.demo9.test9.S9.m1())
  2. 我是m1方法
案例2

定义注解时未使用@Inherited,说明子类无法继承父类上的注解,这个案例中我们将定义一个这样的注解,将注解放在目标类的父类上,来看一下效果。

定义注解Ann10
 
  1. package com.javacode2018.aop.demo9.test10;
  2. import java.lang.annotation.*;
  3. @Retention(RetentionPolicy.RUNTIME)
  4. @Target(ElementType.TYPE)
  5. @Inherited
  6. public @interface Ann10 {
  7. }
来2个父子类

注意:

S10Parent为父类,并且使用了Anno10注解,内部定义了2个方法大家注意一下

而S10位代理的目标类,继承了S10Parent,内部重写了父类的m2方法,并且又新增了一个m3方法

 
  1. package com.javacode2018.aop.demo9.test10;
  2. @Ann10
  3. class S10Parent {
  4. public void m1() {
  5. System.out.println("我是S10Parent.m1()方法");
  6. }
  7. public void m2() {
  8. System.out.println("我是S10Parent.m2()方法");
  9. }
  10. }
  11. public class S10 extends S10Parent {
  12. @Override
  13. public void m2() {
  14. System.out.println("我是S10.m2()方法");
  15. }
  16. public void m3() {
  17. System.out.println("我是S10.m3()方法");
  18. }
  19. }
来个Aspect类
 
  1. package com.javacode2018.aop.demo9.test10;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest10 {
  8. //匹配目标方法声明的类上有@Anno10注解
  9. @Pointcut("@within(com.javacode2018.aop.demo9.test10.Ann10)")
  10. public void pc() {
  11. }
  12. @Before("pc()")
  13. public void beforeAdvice(JoinPoint joinPoint) {
  14. System.out.println(joinPoint);
  15. }
  16. }
测试用例

S10为目标类,依次执行代理对象的m1、m2、m3方法,最终会调用目标类target中对应的方法。

 
  1. @Test
  2. public void test10() {
  3. S10 target = new S10();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest10.class);
  7. S10 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. proxy.m2();
  10. proxy.m3();
  11. }
运行输出
 
  1. execution(void com.javacode2018.aop.demo9.test10.S10Parent.m1())
  2. 我是S10Parent.m1()方法
  3. 我是S10.m2()方法
  4. 我是S10.m3()方法
分析结果

从输出中可以看出,只有m1方法被拦截了,其他2个方法没有被拦截。

确实是这样的,m1方法的是由S10Parent定义的,这个类上面有Ann10注解。

而m2方法虽然也在S10Parent中定义了,但是这个方法被子类S10重写了,所以调用目标对象中的m2方法的时候,此时发现m2方法是由S10定义的,而S10.class.getAnnotation(Ann10.class)为空,所以这个方法不会被拦截。

同样m3方法也是S10中定义的,也不会被拦截。

案例3

对案例2进行改造,在注解的定义上面加上@Inherited,此时子类可以继承父类的注解,此时3个方法都会被拦截了。

下面上代码,下面代码为案例2代码的一个拷贝,不同地方只是注解的定义上多了@Inherited

定义注解Ann11
 
  1. import java.lang.annotation.*;
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Target(ElementType.TYPE)
  4. @Inherited
  5. public @interface Ann11 {
  6. }
2个父子类
 
  1. package com.javacode2018.aop.demo9.test11;
  2. @Ann11
  3. class S11Parent {
  4. public void m1() {
  5. System.out.println("我是S11Parent.m1()方法");
  6. }
  7. public void m2() {
  8. System.out.println("我是S11Parent.m2()方法");
  9. }
  10. }
  11. public class S11 extends S11Parent {
  12. @Override
  13. public void m2() {
  14. System.out.println("我是S11.m2()方法");
  15. }
  16. public void m3() {
  17. System.out.println("我是S11.m3()方法");
  18. }
  19. }
Aspect类
 
  1. package com.javacode2018.aop.demo9.test11;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest11 {
  8. @Pointcut("@within(com.javacode2018.aop.demo9.test11.Ann11)")
  9. public void pc() {
  10. }
  11. @Before("pc()")
  12. public void beforeAdvice(JoinPoint joinPoint) {
  13. System.out.println(joinPoint);
  14. }
  15. }
测试用例
 
  1. @Test
  2. public void test11() {
  3. S11 target = new S11();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest11.class);
  7. S11 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. proxy.m2();
  10. proxy.m3();
  11. }
运行输出
 
  1. execution(void com.javacode2018.aop.demo9.test11.S11Parent.m1())
  2. 我是S11Parent.m1()方法
  3. execution(void com.javacode2018.aop.demo9.test11.S11.m2())
  4. 我是S11.m2()方法
  5. execution(void com.javacode2018.aop.demo9.test11.S11.m3())
  6. 我是S11.m3()方法

这次3个方法都被拦截了。

7、@target

用法

@target(注解类型):判断目标对象target类型上是否有指定的注解;@target中注解类型也必须是全限定类型名。

匹配规则
 
  1. target.class.getAnnotation(指定的注解类型) != null

2种情况可以匹配

  • 注解直接标注在目标类上
  • 注解标注在父类上,但是注解必须是可以继承的,即定义注解的时候,需要使用@Inherited标注
案例1

注解直接标注在目标类上,这种情况目标类会被匹配到。

自定义一个注解Ann6
 
  1. package com.javacode2018.aop.demo9.test6;
  2. import java.lang.annotation.ElementType;
  3. import java.lang.annotation.Retention;
  4. import java.lang.annotation.RetentionPolicy;
  5. import java.lang.annotation.Target;
  6. @Retention(RetentionPolicy.RUNTIME)
  7. @Target(ElementType.TYPE)
  8. public @interface Ann6 {
  9. }
目标类S6上直接使用@Ann1
 
  1. package com.javacode2018.aop.demo9.test6;
  2. @Ann6
  3. public class S6 {
  4. public void m1() {
  5. System.out.println("我是m1");
  6. }
  7. }
来个Aspect类
 
  1. package com.javacode2018.aop.demo9.test6;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest6 {
  8. //@1:目标类上有@Ann1注解
  9. @Pointcut("@target(Ann1)")
  10. public void pc() {
  11. }
  12. @Before("pc()")
  13. public void beforeAdvice(JoinPoint joinPoint) {
  14. System.out.println(joinPoint);
  15. }
  16. }
测试代码
 
  1. @Test
  2. public void test6() {
  3. S6 target = new S6();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest6.class);
  7. S6 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. System.out.println("目标类上是否有 @Ann6 注解:" + (target.getClass().getAnnotation(Ann6.class) != null));
  10. }
运行输出
 
  1. execution(void com.javacode2018.aop.demo9.test6.S6.m1())
  2. 我是m1
  3. 目标类上是否有 @Ann6 注解:true
案例2

注解标注在父类上,注解上没有@Inherited,这种情况下,目标类无法匹配到,下面看代码

注解Ann7
 
  1. package com.javacode2018.aop.demo9.test7;
  2. import java.lang.annotation.ElementType;
  3. import java.lang.annotation.Retention;
  4. import java.lang.annotation.RetentionPolicy;
  5. import java.lang.annotation.Target;
  6. @Retention(RetentionPolicy.RUNTIME)
  7. @Target(ElementType.TYPE)
  8. public @interface Ann7 {
  9. }
来2个父子类,父类上有@Ann7,之类S7为目标类
 
  1. package com.javacode2018.aop.demo9.test7;
  2. import java.lang.annotation.Target;
  3. @Ann7
  4. class S7Parent {
  5. }
  6. public class S7 extends S7Parent {
  7. public void m1() {
  8. System.out.println("我是m1");
  9. }
  10. public static void main(String[] args) {
  11. System.out.println(S7.class.getAnnotation(Target.class));
  12. }
  13. }
来个Aspect类
 
  1. package com.javacode2018.aop.demo9.test7;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest7 {
  8. /**
  9. * 匹配目标类上有Ann7注解
  10. */
  11. @Pointcut("@target(com.javacode2018.aop.demo9.test7.Ann7)")
  12. public void pc() {
  13. }
  14. @Before("pc()")
  15. public void beforeAdvice(JoinPoint joinPoint) {
  16. System.out.println(joinPoint);
  17. }
  18. }
测试代码
 
  1. @Test
  2. public void test7() {
  3. S7 target = new S7();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest7.class);
  7. S7 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. System.out.println("目标类上是否有 @Ann7 注解:" + (target.getClass().getAnnotation(Ann7.class) != null));
  10. }
运行输出
 
  1. 我是m1
  2. 目标类上是否有 @Ann7 注解:false
分析结果

@Ann7标注在了父类上,但是@Ann7定义的时候没有使用@Inherited,说明之类无法继承父类上面的注解,所以上面的目标类没有被拦截,下面我们将@Ann7的定义改一下,加上@Inherited

 
  1. package com.javacode2018.aop.demo9.test7;
  2. import java.lang.annotation.*;
  3. @Retention(RetentionPolicy.RUNTIME)
  4. @Target(ElementType.TYPE)
  5. @Inherited
  6. public @interface Ann7 {
  7. }
再次运行输出
 
  1. execution(void com.javacode2018.aop.demo9.test7.S7.m1())
  2. 我是m1
  3. 目标类上是否有 @Ann7 注解:true

此时目标对象被拦截了。

8、@args

用法

@args(注解类型):方法参数所属的类上有指定的注解;注意不是参数上有指定的注解,而是参数类型的类上有指定的注解。

案例1
 
  1. @Pointcut("@args(Ann8)"):匹配方法只有一个参数,并且参数所属的类上有Ann8注解

可以匹配下面的代码,m1方法的第一个参数类型是Car类型,Car类型上有注解Ann8

 
  1. @Ann8
  2. class Car {
  3. }
  4. public void m1(Car car) {
  5. System.out.println("我是m1");
  6. }
案例2
 
  1. @Pointcut("@args(*,Ann8)"):匹配方法只有2个参数,且第2个参数所属的类型上有Ann8注解

可以匹配下面代码

 
  1. @Ann8
  2. class Car {
  3. }
  4. public void m1(String name,Car car) {
  5. System.out.println("我是m1");
  6. }
案例3
 
  1. @Pointcut("@args(..,com.javacode2018.aop.demo9.test8.Ann8)"):匹配参数数量大于等于1,且最后一个参数所属的类型上有Ann8注解
  2. @Pointcut("@args(*,com.javacode2018.aop.demo9.test8.Ann8,..)"):匹配参数数量大于等于2,且第2个参数所属的类型上有Ann8注解
  3. @Pointcut("@args(..,com.javacode2018.aop.demo9.test8.Ann8,*)"):匹配参数数量大于等于2,且倒数第2个参数所属的类型上有Ann8注解

这个案例代码,大家自己写一下,体验一下。

9、@annotation

用法

@annotation(注解类型):匹配被调用的方法上有指定的注解。

案例
定义一个注解,可以用在方法上
 
  1. package com.javacode2018.aop.demo9.test12;
  2. import java.lang.annotation.ElementType;
  3. import java.lang.annotation.Retention;
  4. import java.lang.annotation.RetentionPolicy;
  5. import java.lang.annotation.Target;
  6. @Retention(RetentionPolicy.RUNTIME)
  7. @Target(ElementType.METHOD)
  8. public @interface Ann12 {
  9. }
定义2个类

S12Parent为父类,内部定义了2个方法,2个方法上都有@Ann12注解

S12是代理的目标类,也是S12Parent的子类,内部重写了m2方法,重写之后m2方法上并没有@Ann12注解,S12内部还定义2个方法m3和m4,而m3上面有注解@Ann12

 
  1. package com.javacode2018.aop.demo9.test12;
  2. class S12Parent {
  3. @Ann12
  4. public void m1() {
  5. System.out.println("我是S12Parent.m1()方法");
  6. }
  7. @Ann12
  8. public void m2() {
  9. System.out.println("我是S12Parent.m2()方法");
  10. }
  11. }
  12. public class S12 extends S12Parent {
  13. @Override
  14. public void m2() {
  15. System.out.println("我是S12.m2()方法");
  16. }
  17. @Ann12
  18. public void m3() {
  19. System.out.println("我是S12.m3()方法");
  20. }
  21. public void m4() {
  22. System.out.println("我是S12.m4()方法");
  23. }
  24. }
来个Aspect类

当被调用的目标方法上有@Ann12注解的时,会被beforeAdvice处理。

 
  1. package com.javacode2018.aop.demo9.test12;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. @Aspect
  7. public class AspectTest12 {
  8. @Pointcut("@annotation(com.javacode2018.aop.demo9.test12.Ann12)")
  9. public void pc() {
  10. }
  11. @Before("pc()")
  12. public void beforeAdvice(JoinPoint joinPoint) {
  13. System.out.println(joinPoint);
  14. }
  15. }
测试用例

S12作为目标对象,创建代理,然后分别调用4个方法

 
  1. @Test
  2. public void test12() {
  3. S12 target = new S12();
  4. AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
  5. proxyFactory.setTarget(target);
  6. proxyFactory.addAspect(AspectTest12.class);
  7. S12 proxy = proxyFactory.getProxy();
  8. proxy.m1();
  9. proxy.m2();
  10. proxy.m3();
  11. proxy.m4();
  12. }
运行输出
 
  1. execution(void com.javacode2018.aop.demo9.test12.S12Parent.m1())
  2. 我是S12Parent.m1()方法
  3. 我是S12.m2()方法
  4. execution(void com.javacode2018.aop.demo9.test12.S12.m3())
  5. 我是S12.m3()方法
  6. 我是S12.m4()方法
分析结果

m1方法位于S12Parent中,上面有@Ann12注解,被拦截了,m3方法上有@Ann12注解,被拦截了,而m4上没有@Ann12注解,没有被拦截,这3个方法的执行结果都很容易理解。

重点在于m2方法的执行结果,没有被拦截,m2方法虽然在S12Parent中定义的时候也有@Ann12注解标注,但是这个方法被S1给重写了,在S1中定义的时候并没有@Ann12注解,代码中实际上调用的是S1中的m2方法,发现这个方法上并没有@Ann12注解,所以没有被拦截。

10、bean

用法

bean(bean名称):这个用在spring环境中,匹配容器中指定名称的bean。

案例
来个类BeanService
 
  1. package com.javacode2018.aop.demo9.test13;
  2. public class BeanService {
  3. private String beanName;
  4. public BeanService(String beanName) {
  5. this.beanName = beanName;
  6. }
  7. public void m1() {
  8. System.out.println(this.beanName);
  9. }
  10. }
来个Aspect类
 
  1. package com.javacode2018.aop.demo9.test13;
  2. import org.aspectj.lang.JoinPoint;
  3. import org.aspectj.lang.annotation.Aspect;
  4. import org.aspectj.lang.annotation.Before;
  5. import org.aspectj.lang.annotation.Pointcut;
  6. import org.springframework.stereotype.Component;
  7. @Aspect
  8. public class Aspect13 {
  9. //拦截spring容器中名称为beanService2的bean
  10. @Pointcut("bean(beanService2)")
  11. public void pc() {
  12. }
  13. @Before("pc()")
  14. public void beforeAdvice(JoinPoint joinPoint) {
  15. System.out.println(joinPoint);
  16. }
  17. }
来个spring配置类
 
  1. package com.javacode2018.aop.demo9.test13;
  2. import org.springframework.context.annotation.Bean;
  3. import org.springframework.context.annotation.Configuration;
  4. import org.springframework.context.annotation.EnableAspectJAutoProxy;
  5. @Configuration
  6. @EnableAspectJAutoProxy // 这个可以启用通过AspectJ方式自动为符合条件的bean创建代理
  7. public class MainConfig13 {
  8. //将Aspect13注册到spring容器
  9. @Bean
  10. public Aspect13 aspect13() {
  11. return new Aspect13();
  12. }
  13. @Bean
  14. public BeanService beanService1() {
  15. return new BeanService("beanService1");
  16. }
  17. @Bean
  18. public BeanService beanService2() {
  19. return new BeanService("beanService2");
  20. }
  21. }

这个配置类中有个@EnableAspectJAutoProxy,这个注解大家可能比较陌生,这个属于aop中自动代理的范围,后面会有文章详细介绍这块,这里大家暂时先不用关注。

测试用例

下面启动spring容器,加载配置类MainConfig13,然后分别获取beanService1和beanService2,调用他们的m1方法,看看效果

 
  1. @Test
  2. public void test13() {
  3. AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MainConfig13.class);
  4. //从容器中获取beanService1
  5. BeanService beanService1 = context.getBean("beanService1", BeanService.class);
  6. beanService1.m1();
  7. //从容器中获取beanService2
  8. BeanService beanService2 = context.getBean("beanService2", BeanService.class);
  9. beanService2.m1();
  10. }
运行输出
 
  1. beanService1
  2. execution(void com.javacode2018.aop.demo9.test13.BeanService.m1())
  3. beanService2

beanService2的m1方法被拦截了。

11、reference pointcut

表示引用其他命名切入点。

有时,我们可以将切入专门放在一个类中集中定义。

其他地方可以通过引用的方式引入其他类中定义的切入点。

语法如下:

 
  1. @Pointcut("完整包名类名.方法名称()")

若引用同一个类中定义切入点,包名和类名可以省略,直接通过方法就可以引用。

比如下面,我们可以将所有切入点定义在一个类中

 
  1. package com.javacode2018.aop.demo9.test14;
  2. import org.aspectj.lang.annotation.Pointcut;
  3. public class AspectPcDefine {
  4. @Pointcut("bean(bean1)")
  5. public void pc1() {
  6. }
  7. @Pointcut("bean(bean2)")
  8. public void pc2() {
  9. }
  10. }

下面顶一个一个Aspect类,来引用上面的切入点

 
  1. package com.javacode2018.aop.demo9.test14;
  2. import org.aspectj.lang.annotation.Aspect;
  3. import org.aspectj.lang.annotation.Pointcut;
  4. @Aspect
  5. public class Aspect14 {
  6. @Pointcut("com.javacode2018.aop.demo9.test14.AspectPcDefine.pc1()")
  7. public void pointcut1() {
  8. }
  9. @Pointcut("com.javacode2018.aop.demo9.test14.AspectPcDefine.pc1() || com.javacode2018.aop.demo9.test14.AspectPcDefine.pc2()")
  10. public void pointcut2() {
  11. }
  12. }

12、组合型的pointcut

Pointcut定义时,还可以使用&&、||、!运算符。

  • &&:多个匹配都需要满足
  • ||:多个匹配中只需满足一个
  • !:匹配不满足的情况下
 
  1. @Pointcut("bean(bean1) || bean(bean2)") //匹配bean1或者bean2
  2. @Pointcut("@target(Ann1) && @Annotation(Ann2)") //匹配目标类上有Ann1注解并且目标方法上有Ann2注解
  3. @Pointcut("@target(Ann1) && !@target(Ann2)") // 匹配目标类上有Ann1注解但是没有Ann2注解

总结

本文详解了@Pointcut的12种用法,案例大家一定要敲一遍,敲的过程中,会遇到问题,然后解决问题,才能够加深理解。

有问题的也欢迎大家留言交流,谢谢!

案例源码

 
  1. https://gitee.com/javacode2018/spring-series

本博客所有系列案例代码以后都会放到这个上面,大家watch一下,可以持续关注动态。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值