美文网首页
Spring之AOP【二】

Spring之AOP【二】

作者: zhanghTK | 来源:发表于2018-07-01 17:40 被阅读0次

    title: Spring之AOP【二】
    date: 2017-03-25 02:42:16
    tags:

    • Java
    • Spring
      categories: Spring

    @AspectJ形式的Pointcut

    Pointcut的组成:

    • Pointcut Expression
      • 以@Pointcut为载体
      • 由两部分组成:Pointcut标识符,表达式匹配模式
    • Pointcut Signature
      • Pointcut Expression的载体

    Pointcut标识符

    execution

    日常使用最多的标识符,使用execution标识符的Pointcut表达式格式:

    execution (modifiers-pattern? ret-type-pattern declaring-type-pattern? 
                name-pattern(param-pattern) throws-pattern?)
    
    • 方法返回类型、方法名以及参数必须制定,其他可以省略
    • 可选通配符:*..
      • *:匹配一个单词
      • ..:只能在declaring-type-pattern和param-pattern位置使用
        • 用于declaring-type-pattern可以指定多个层次
        • 用于param-pattern表示可以有0到多个参数,可以与*和具体类型组合

    within

    指定类型,类型下所有方法。可以使用*..扩展,like:within(tk.zhangh.spring..*)

    this和target

    • this:目标对象的代理对象
    • target:目标对象

    Spring中使用this和target实际作用类似

    args

    指定参数类型,指定参数数量

    与execution标识符不同,args标识符会在运行期间动态检查参数类型

    @within

    指定类型,类型下的所有方法,要求类型标记了指定注解,like:

    @Retention(RetentionPolicy.RUNTIME)
    @Target({ElementType.METHOD, ElementType.TYPE})
    public @Log AnyJoinpontAnnotation{}
    
    @Log
    public class Bean {
      public void method();
    }
    
    @with(Log)
    

    @target

    指定标记了给定注解类型的目标对象的所有方法

    @args

    指定参数类型,要求参数参数类型标记了指定注解

    @annotation

    指定标记了指定注解的方法,@Transctional的实现方式

    Pointcut表达式的解析

    所有@AspectJ形式声明的这些Pointcut表达式最终都会转化成Pointcut的具体实现。

    AspectJExpressionPointcut如同他的名字面向AspectJ的pointcut实现,整个继承体系:


    pointcut.png

    @AspectJ形式的Advice

    使用@Aspect注解标记的类中,具体的Advice形式由具体的Advice注解标示。

    注解的方法中需要访问上下文信息最主要的方式:将方法的第一个参数声明为JoinPoint类型

    • @Before

    • @AfterReturning

    • @AfterThrowing

      • 获取异常:在JoinPoint类型参数后面加上RuntimeException类型参数
    • @After

    • @Around

      • 获取上下文信息不同以上,需要方法第一个参数声明为ProcessingJoinPoint类型
    • @DeclareParents

      • 最特殊,使用如下:
      @Aspect
      public class IntroductionAspect {
        @DeclareParents(value="....NewImpl", defaultImpl=Target.class)
        public INew new;
      }
      

    其他

    Advice执行顺序

    • 同一个Aspect中最先声明的Advice拥有最高优先级,优先入栈
    • 不同Aspect的Advice需要实现Order接口声明优先级

    同一对象内的嵌套方法调用拦截失效

    以事务为例,事务管理也是使用AOP,具体是@annotation形式的Pointcut声明(这样我就不用声明Advice了)

    public class ServiceImpl implents Service {
      public void methodA() {
        doSomething();
        methodB();
      }
      
      @Transactional
      public void methodB() {
        // ...
      }
    }
    

    当在aFun内调用bFun时事务没有开启,也就是AOP没有生效,原因:


    AOP嵌套调用.png

    我们期望虚线的调用方式,实际上调用时红色的路线,添加在代理对象上的AOP逻辑在嵌套调用时根本没有机会触发。在事务处理时尤其要注意避免这样的嵌套调用问题。

    解决:

    • 使用AopContext.currentProxy()获取代理对象
    • 从ApplicationContext中获取代理对象

    不管是那种方式都要注入相关Bean,具体那种更优雅由你来决定了。

    相关文章

      网友评论

          本文标题:Spring之AOP【二】

          本文链接:https://www.haomeiwen.com/subject/vadseftx.html