AOP相关概念
在学习AOP实现原理之前,先了解下AOP相关基础知识。
AOP面向切面编程,它可以通过预编译方式或者基于动态代理对我们编写的代码进行拦截(也叫增强处理),在方法执行前后可以做一些操作,一般我们会看到以下几个概念:
连接点(JointPoint): AOP进行切入的位置称为连接点,一般指程序中的某个方法,对该方法进行拦截
通知(Advice): 在某个连接点执行的操作称为通知,也就是被拦截方法执行前后需要执行的操作称为通知,一共有五种
- 前置通知:作用于被拦截方法执行之前
- 后置通知:作用于被拦截方法执行之后进行的操作,无论被拦截方法是否抛出异常都会执行
- 环绕通知:作用于被拦截方法执行之前和执行之后
- 返回通知:作用于被拦截方法正常执行完毕返回时,如果抛出异常将不会执行
- 异常通知:作用于被拦截方法抛出异常时
切点(Pointcut): 切点作用在于让程序知道需要在哪个连接点(方法)上执行通知,所以它也可以是一个表达式,匹配所有需要拦截的方法。
切面(Aspect): 切点和通知共同组成了切面,其中切点定义了需要在哪些连接点上执行通知,通知里面定义了具体需要进行的操作。
织入(Weaving):将切面连接到应用程序类型或者对象上,创建一个被通知的对象(advised object)的过程称为织入,换句话说织入就是将切面应用到目标对象的过程,它可以在编译期时(使用AspectJ)、加载时或者在运行时实现,Spring AOP是在运行时基于动态代理实现的。
Advisor:它是对切面的封装,使用了@AspectJ注解的类会被封装成Advisor。
Spring AOP和AspectJ区别
Spring AOP
Spring AOP是基于动态代理实现拦截功能的,默认使用JDK动态代理实现,当然这需要目标对象实现接口,如果目标对象没有实现接口,则使用CGLIB生成。
AspectJ
AspectJ提供了三种方式实现AOP:
编译时织入:在编译期间将代码进行织入到目标类的class文件中。
编译后织入:在编译后将代码织入到目标类的class文件中。
加载时织入:在JVM加载class文件的时候进行织入。
Spring AOP的应用
了解了AOP相关知识后我们来实现一个需求:
- 自定义一个日志注解MyLogger
- 对使用了MyLogger注解的方法进行拦截,在方法的执行前后分别进行一些操作(环绕通知):
- 方法执行前打印方法传入的参数
- 方法执行后打印方法的返回值
自定义注解
1 | import java.lang.annotation.*; |
定义切面Aspect
这里使用注解@Aspect来标记这是一个切面,切面是切点和通知的集合,分别使用注解@Pointcut和@Around实现。
1 | 4j |
切点Pointcut
使用表达式@annotation(com.demo.mybatis.annotation.MyLogger)
匹配所有使用了@MyLogger注解的方法。
1 | /** |
通知Advice
定义一个logAroudAdvice
方法,使用@Around注解标记这是一个环绕通知,logPoiontcut()
引用了切点,表示通知要作用于哪些连接点上,该方法需要传入一个ProceedingJoinPoint类型参数(连接点):
1 | /** |
完整的切面如下:
1 | import lombok.extern.slf4j.Slf4j; |
测试
定义一个用于计算的Service,实现一个两数相加的方法addTwoNum
,并使用@MyLogger
注解,对方法进行拦截,在方法执行前后打印相关日志
1 | 4j |
编写单元测试:
1 |
|
由于ComputeService没有实现接口,可以看到Spring默认使用了CGLIB生成对象:
日志输出如下,可以看到方法执行前后打印了相关日志:
1 | 开始执行方法:addTwoNum |
参考