Spring AOP使用整理:使用@AspectJ风格的切面声明

xiaoxiao2022-06-11  36

要启用基于@AspectJ风格的切面声明,需要进行以下的配置:

<!-- 启用@AspectJ风格的切面声明 --> <aop:aspectj-autoproxy proxy-target-class="true"/> <!-- 通过注解定义bean。默认同时也通过注解自动注入 --> <context:component-scan base-package="com.cjm"/>

 

基于@AspectJ风格的切面声明的源码:

/** * 声明本类为一个切面 */ @Component @Aspect public class MyAspectJ { /** * 声明一个切入点(包括切入点表达式和切入点签名) */ @Pointcut("execution(* com.cjm.model..*.*(..))") public void pointcut1(){} /** * 声明一个前置通知 */ @Before("pointcut1()") public void beforeAdvide(JoinPoint point){ System.out.println("触发了前置通知!"); } /** * 声明一个后置通知 */ @After("pointcut1()") public void afterAdvie(JoinPoint point){ System.out.println("触发了后置通知,抛出异常也会被触发!"); } /** * 声明一个返回后通知 */ @AfterReturning(pointcut="pointcut1()", returning="ret") public void afterReturningAdvice(JoinPoint point, Object ret){ System.out.println("触发了返回后通知,抛出异常时不被触发,返回值为:" + ret); } /** * 声明一个异常通知 */ @AfterThrowing(pointcut="pointcut1()", throwing="throwing") public void afterThrowsAdvice(JoinPoint point, RuntimeException throwing){ System.out.println("触发了异常通知,抛出了RuntimeException异常!"); } /** * 声明一个环绕通知 */ @Around("pointcut1()") public Object aroundAdvice(ProceedingJoinPoint point)throws Throwable{ System.out.println("触发了环绕通知 开始"); Object o = point.proceed(); System.out.println("触发了环绕通知 结束"); return o; } }

 

1、切入点表达式的格式:execution([可见性] 返回类型 [声明类型].方法名(参数) [异常])

 

2、切入点表达式通配符:      *:匹配所有字符      ..:一般用于匹配多个包,多个参数      +:表示类及其子类

 

3、切入点表达式支持逻辑运算符:&&、||、!

 

4、切入点表达式关键词:      1)execution:用于匹配子表达式。

            //匹配com.cjm.model包及其子包中所有类中的所有方法,返回类型任意,方法参数任意            @Pointcut("execution(* com.cjm.model..*.*(..))")            public void before(){}

 

      2)within:用于匹配连接点所在的Java类或者包。

            //匹配Person类中的所有方法            @Pointcut("within(com.cjm.model.Person)")            public void before(){}

 

            //匹配com.cjm包及其子包中所有类中的所有方法

            @Pointcut("within(com.cjm..*)")            public void before(){}

 

     3) this:用于向通知方法中传入代理对象的引用。            @Before("before() && this(proxy)")            public void beforeAdvide(JoinPoint point, Object proxy){                  //处理逻辑            }

 

      4)target:用于向通知方法中传入目标对象的引用。            @Before("before() && target(target)            public void beforeAdvide(JoinPoint point, Object proxy){                  //处理逻辑            }

 

      5)args:用于将参数传入到通知方法中。            @Before("before() && args(age,username)")            public void beforeAdvide(JoinPoint point, int age, String username){                  //处理逻辑            }       6)@within:用于匹配在类一级使用了参数确定的注解的类,其所有方法都将被匹配。 

            @Pointcut("@within(com.cjm.annotation.AdviceAnnotation)") - 所有被@AdviceAnnotation标注的类都将匹配            public void before(){}

 

@Retention(RetentionPolicy.RUNTIME) @Target({ElementType.TYPE, ElementType.METHOD}) @Documented @Inherited public @interface AdviceAnnotation { }

  

      7)@target:和@within的功能类似,但必须要指定注解接口的保留策略为RUNTIME。            @Pointcut("@target(com.cjm.annotation.AdviceAnnotation)")            public void before(){}

 

      8)@args:传入连接点的对象对应的Java类必须被@args指定的Annotation注解标注。            @Before("@args(com.cjm.annotation.AdviceAnnotation)")            public void beforeAdvide(JoinPoint point){                  //处理逻辑            }

  

public class Person { public void say(Address address){ //处理逻辑 } } @AdviceAnnotation public class Address { }

 

  如果需要在Person类的say方法被调用时触发beforeAdvide通知,那么say方法的参数对应的Java类型Address类必须要被@AdviceAnnotation标注。

 

      9)@annotation:匹配连接点被它参数指定的Annotation注解的方法。也就是说,所有被指定注解标注的方法都将匹配。            @Pointcut("@annotation(com.cjm.annotation.AdviceAnnotation)")            public void before(){}

 

public class Person { @AdviceAnnotation public void say(Address address){ //处理逻辑 } }

  

  Person类的say方法被@AdviceAnnotation标注,所以它匹配。

 

      10)bean:通过受管Bean的名字来限定连接点所在的Bean。该关键词是Spring2.5新增的。            @Pointcut("bean(person)")            public void before(){}

 

            id为person的受管Bean中的所有方法都将匹配。

相关资源:Java 面经手册·小傅哥(公众号:bugstack虫洞栈).pdf
转载请注明原文地址: https://www.6miu.com/read-4931917.html

最新回复(0)