一、AOP概念
Spring AOP中自我调用的问题
Spring之AOP详解
在开始之前,让我们对术语和核心概念进行回顾:
- 切面(Aspect) – 一些横跨多个类的公共模块,如日志、安全、事务等。简单地说,日志模块就是一个切面。
- 通知(Advice) – AOP 框架中的增强处理。通知描述了切面何时执行以及如何执行增强处理。
- 连接点(Joint Point) – 目标类中插入代码的地方。连接点可以是方法、异常、字段,连接点处的切面代码会在方法执行、异常抛出、字段修改时触发执行。
- 建议(Advice) – 在连接点插入的实际代码(即切面的方法),有5种不同类型(后面介绍)。
- 切点(Pointcut) – 定义了连接点的条件,一般通过正则表达式。例如,可以定义所有以
loadUser
开头的方法作为连接点,插入日志代码。 - 织入(Weaving) – 将增强处理添加到目标对象中,并创建一个被增强的对象。
通知类型
- before – 在方法之前运行建议(插入的代码)
- after – 不管方法是否成功执行,在方法之后运行插入建议(插入的代码)
- after-returning – 当方法执行成功,在方法之后运行建议(插入的代码)
- after-throwing – 仅在方法抛出异常后运行建议(插入的代码)
- around – 在方法被调用之前和之后运行建议(插入的代码)
AspectJ和Spring AOP都使用不同类型的编织,这会影响它们在性能和易用性方面的行为。
AspectJ使用三种不同的编织方式:
- 编译时编织:AspectJ编译器将方面和应用程序的源代码都作为输入,并生成编织类文件作为输出
- 编译后编织:这也称为二进制编织。它用于与我们的方面编织现有的类文件和JAR文件
- 加载时编织:这与以前的二进制编织完全一样,不同之处在于编织被推迟到类加载器将类文件加载到JVM为止
由于AspectJ使用编译时编织和类加载时编织,因此Spring AOP使用了运行时编织。
通过运行时编织,可以在应用程序执行期间使用目标对象的代理来编织各方面–使用JDK动态代理或CGLIB代理:
Spring AOP默认将标准JDK动态代理用于AOP代理。这使得可以代理任何接口(或一组接口)。
Spring AOP也可以使用CGLIB代理。这对于代理类而不是接口是必需的。如果业务对象未实现接口,则默认情况下使用CGLIB。
在那些需要建议在接口上未声明的方法或需要将代理对象作为具体类型传递给方法的情况下(在极少数情况下),可以 强制使用CGLIB。
使用方式
Spring 2.0 之后,Spring AOP有了两种配置方式。
-
schema-based:Spring 2.0 以后使用 XML 的方式来配置,使用 命名空间
<aop/>
-
@AspectJ 配置:Spring 2.0 以后提供的注解方式。这里虽然叫做 @AspectJ,但是这个和 AspectJ 其实没啥关系。