SpringAOP中的通知Advice详解
一、概述
AOP 中的通知是基于连接点(Join point)业务逻辑的一种增强,Spring AOP 提供了下面五种通知类型:
- Before advice(前置通知):连接点前面执行,不能终止后续流程,除非抛异常
- After returning advice(后置通知):连接点正常返回时执行,有异常不执行
- Around advice(环绕通知):围绕连接点前后执行,也能捕获异常处理
- After advice(最终通知):连接点退出时执行,无论是正常退出还是异常退出
- After throwing advice(异常通知):连接点方法抛出异常时执行
AOP 的连接点一般是指目标类的方法,五种通知类型执行的节点如下:
二、通知的定义
Spring AOP 可以基于 XML 方式和基于注解方式定义,只是写法不同,这里只使用注解的方式来讲解通知的详细用法。
1. 前置通知
在 @Aspect
切面类中使用 @Before
注解简单地定义一个前置通知。
@Aspect @Component public class DemoAspect { @Before("execution(* cn.codeartist.spring.aop.advice.*.*(..))") public void doBefore() { // 自定义逻辑 } }
2. 后置通知
方法正常返回,会执行后置通知,使用 @AfterReturning
注解定义后置通知。
@AfterReturning("execution(* cn.codeartist.spring.aop.advice.*.*(..))") public void doAfterReturning() { // 自定义逻辑 }
注解的 returning
属性可以绑定目标方法返回值,用于在通知中获取目标方法执行完成后的返回结果。
@AfterReturning(pointcut = "pointcut()", returning = "retVal") public void doAfterReturning(Object retVal) { // 自定义逻辑(通过参数绑定方法切入点方法的返回值) }
当注解使用了 returning
属性时,切入点会增加返回值类型的限制,上面使用的 Object
类型可以匹配到所有返回值类型的目标方法。
例如下面的情况,后置通知的代码不会被执行:
// 目标方法 public String doService() { return "码匠公众号"; } // 后置通知定义 @AfterReturning(pointcut = "pointcut()", returning = "retVal") public void doAfterReturning(Integer retVal) { // 目标方法返回值是String类型,结果参数绑定类型是Integer,该通知不会被执行 }
3. 环绕通知
环绕通知可以在方法执行的任何节点添加逻辑,它可以实现另外 4 种通知的功能。
如果需要以线程安全的方式在方法执行前后共享状态,可以使用环绕通知。
用 @Around
注解来定义环绕通知,需要使用 ProceedingJoinPoint
作为参数,来执行目标方法调用。
@Around("execution(* cn.codeartist.spring.aop.advice.*.*(..))") public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable { // 方法执行前逻辑 Object retVal = joinPoint.proceed(); // 方法执行后逻辑 return retVal; }
joinPoint.proceed()
会调用目标方法,或者是调用另一个切面。
虽然环绕通知可以实现另外几种通知的功能,但在使用中都能实现功能的情况下,优先使用其他通知方式。
4. 最终通知
最终通知在方法退出的时候执行,使用 @After
注解定义,最终通知在方法正常退出和抛出异常时都会执行,通常用于资源的释放。
@After("execution(* cn.codeartist.spring.aop.advice.*.*(..))") public void doAfter() { // 自定义逻辑 }
5. 异常通知
方法抛出异常的时候会执行异常通知,使用 @AfterThrowing
定义异常通知。
@AfterThrowing("execution(* cn.codeartist.spring.aop.advice.*.*(..))") public void doAfterThrowing() { // 自定义逻辑 }
注解的 throwing
属性用来绑定目标方法抛出的异常,用于在通知中获取目标方法抛出的异常实例。
@AfterThrowing(pointcut = "pointcut()", throwing = "ex") public void doAfterThrowing(Throwable ex) { // 自定义逻辑(通过参数绑定方法切入点方法抛出的异常) }
当注解使用了 throwing
属性时,切入点会增加异常类型的限制,上面使用的 Throwable
类型可以匹配到所有异常类型。
例如下面的情况,异常通知的代码不会被执行:
// 目标方法 public void doServiceThrow() { throw new RuntimeException("Test exception"); } // 异常通知定义 @AfterThrowing(pointcut = "pointcut()", throwing = "ex") public void doAfterThrowing(NullPointerException ex) { // 目标方法抛出的是RuntimeException,参数绑定类型是NullPointerException,该通知不会被执行 }
三、通知的参数
在定义通知的方法签名上可以指定参数来绑定目标方法的一些信息(例如前面讲到的后置通知和异常通知)。
1. 切入点
在定义通知方法的时候,一般可以使用 JoinPoint
作为参数,环绕通知使用 ProceedingJoinPoint
。常用接口方法如下:
JoinPoint
public interface JoinPoint { // 获取代理对象 Object getThis(); // 获取目标对象 Object getTarget(); // 获取连接点方法的参数 Object[] getArgs(); // 获取连接点方法的签名 Signature getSignature(); }
ProceedingJoinPoint
public interface ProceedingJoinPoint extends JoinPoint { // 执行下一个切面的通知或者目标方法 public Object proceed() throws Throwable; // 执行下一个切面的通知或者目标方法(带参数) public Object proceed(Object[] args) throws Throwable; }
切面的切入点一般为方法,所以
Signature
可以转换为MethodSignature
使用。
2. 通知的参数传递
通知的参数可以通过切点表达式 args
来指定,具体用法会在后面切点表达式详解中讲到。
@Before("pointcut() && args(name,..)") public void doBefore(String name) { // 切点表达式增加参数匹配 }
args(name,..)
也会增加切入点的限制,目标方法的参数个数至少为一个,且第一个参数类型为 String
。
四、通知的顺序
Spring AOP 中一个目标类可以被多个切面切入,多个切面也可以切入一个目标类。 使用 @Order
注解来指定切面的优先级,来控制切面的执行顺序。 在注册切面 Bean 的时候指定 @Order
,如下:
@Order(1) @Aspect @Component public class FirstAspect { // ...... }
优先级高的切面先执行,通知执行的顺序如下:
可以得出:
- 优先级高的切面,前置通知先执行
- 优先级低的切面,后置通知先执行
Order
值越小,优先级越大。
Spring AOP 是基于动态代理的拦截器模式实现的,切面模型与拦截器模型相似,如下:
五、附录
1. 常用注解
注解 | 描述 |
@Aspect | 定义切面类 |
@Before | 定义前置通知 |
@AfterReturning | 定义后置通知 |
@Around | 定义环绕通知 |
@After | 定义最终通知 |
@AfterThrowing | 定义异常通知 |
到此这篇关于SpringAOP中的通知Advice详解的文章就介绍到这了,更多相关SpringAOP的通知内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
SpringBoot整合liquibase及liquibase生成初始化脚本的方式
这篇文章主要介绍了SpringBoot整合liquibase的相关资料,文中给大家介绍了liquibase生成初始化脚本的两种方式,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下2022-02-02基于springboot拦截器HandlerInterceptor的注入问题
这篇文章主要介绍了springboot拦截器HandlerInterceptor的注入问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2021-09-09SpringBoot中@Scheduled()注解以及cron表达式详解
这篇文章主要介绍了SpringBoot中@Scheduled()注解以及cron表达式详解,@Scheduled注解是Spring Boot提供的用于定时任务控制的注解,主要用于控制任务在某个指定时间执行,或者每隔一段时间执行,需要的朋友可以参考下2023-08-08Spring的@CrossOrigin注解使用与CrossFilter对象自定义详解
这篇文章主要介绍了Spring的@CrossOrigin注解使用与CrossFilter对象自定义详解,跨域,指的是浏览器不能执行其他网站的脚本,它是由浏览器的同源策略造成的,是浏览器施加的安全限制,所谓同源是指,域名,协议,端口均相同,需要的朋友可以参考下2023-12-12Java中JUC包(java.util.concurrent)下的常用子类
相信大家已经对并发机制中出现的很多的常见知识点进行了总结,下面这篇文章主要给大家介绍了关于Java中JUC包(java.util.concurrent)下的常用子类的相关资料,文中通过图文以及示例代码介绍的非常详细,需要的朋友可以参考下2022-12-12
最新评论