Spring之AOP【二】


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,具体那种更优雅由你来决定了。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 四、基于@AspectJ的AOP Spring除了支持Schema方式配置AOP,还支持注解方式:使用@Aspec...
    聂叼叼阅读 3,251评论 0 0
  • IoC 容器 Bean 的作用域 自定义作用域实现 org.springframework.beans.facto...
    Hsinwong阅读 7,235评论 0 7
  • 基本知识 其实, 接触了这么久的 AOP, 我感觉, AOP 给人难以理解的一个关键点是它的概念比较多, 而且坑爹...
    永顺阅读 12,658评论 5 114
  • 概述 Spring是什么? Spring是一个开源框架,为了解决企业应用开发的复杂性而创建的,但是现在已经不止于企...
    琅筑阅读 4,933评论 2 8
  • 第1分钟:小说世界观 不知何年何月的北京被分成了三个极端的空间:第一空间、第二空间、第三空间,不同空间的人相互隔绝...
    意不适阅读 3,868评论 0 1