一文让你彻底弄懂 AspectJ 切面注解中五种通知注解的执行顺序

小郝不负流年
小郝不负流年   + 关注
2021-05-29 15:45:30   阅读478   评论1

一、前言

本文不讲什么是AOP,什么是切面,不知道的自己百度,本文主要是探究 AspectJ 切面注解中五种通知注解的执行顺序,以便于我们在实际开发过程中游刃有余的实现我们的业务。

首先我们知道切面通知注解有以下五种:

  • @Before: 前置通知, 在方法执行之前执行
  • @After: 后置通知, 在方法执行之后执行
  • @AfterRunning: 返回通知, 在方法返回结果之后执行
  • @AfterThrowing: 异常通知, 在方法抛出异常之后
  • @Around: 环绕通知, 围绕着方法执行

看到上面的描述我们大致也知道了一个执行顺序,但是在正常请求和有异常时具体的执行顺序我详细大家没有经过实测还是不敢确定的。另外如果同一个方法被多个Aspect类拦截呢它的执行顺序是怎样的呢?我想大家心里可能有猜想,但这些都需要我们实实在在的写一个例子来验证它。

二、编写测试代码

Aspect切面类


  1. /** 
  2.  * 测试 AspectJ 切面注解中五种通知注解 
  3.  * 
  4.  * @author Hoscen 
  5.  * @since 2021/5/29 14:36 
  6.  */  
  7. @Aspect  
  8. @Component  
  9. @Slf4j  
  10. @Order(1)  
  11. public class TestAspect {  
  12.   
  13.   
  14.     @Pointcut("execution(* cn.hoscen.test.testAspect.controller.*.*(..))")  
  15.     public void pointcut() {  
  16.     }  
  17.   
  18.     @Before(value = "pointcut()")  
  19.     public void doBefore(JoinPoint jp) {  
  20.         log.info(Thread.currentThread().getName() + "  doBefore");  
  21.     }  
  22.   
  23.     @Around(value = "pointcut()")  
  24.     public Object doAround(ProceedingJoinPoint pjp) throws Throwable {  
  25.         log.info(Thread.currentThread().getName() + "  doAround 1");  
  26.         Object result = pjp.proceed();  
  27.         log.info(Thread.currentThread().getName() + "  doAround 2");  
  28.         return result;  
  29.     }  
  30.   
  31.     @AfterThrowing(value = "pointcut()", throwing = "exception")  
  32.     public void doAfterThrowing(JoinPoint jp, Exception exception) {  
  33.         log.info(Thread.currentThread().getName() + "  doAfterThrowing");  
  34.         log.error("doAfterThrowing",exception);  
  35.     }  
  36.   
  37.     @AfterReturning(value = "pointcut()", returning = "result")  
  38.     public void doAfterReturning(JoinPoint jp, Object result) {  
  39.         log.info(Thread.currentThread().getName() + "  doAfterReturning");  
  40.     }  
  41.   
  42.     @After(value = "pointcut()")  
  43.     public void doAfter(JoinPoint jp) {  
  44.         log.info(Thread.currentThread().getName() + "  doAfter");  
  45.     }  
  46.   
  47. }  

测试Controller


  1. /** 
  2.  * 测试 AspectJ 切面注解中五种通知注解 
  3.  * 
  4.  * @author Hoscen 
  5.  * @since 2021/5/29 14:44 
  6.  */  
  7. @RestController  
  8. @RequestMapping(value = "/api/test", produces = {MediaType.APPLICATION_JSON_UTF8_VALUE})  
  9. public class TestAspectController extends BaseController {  
  10.   
  11.     @RequestMapping(value = "/ok", method = RequestMethod.GET)  
  12.     public BaseReturn getOk() {  
  13.         return BaseReturn.markSuccess(null);  
  14.     }  
  15.   
  16.     @RequestMapping(value = "/error", method = RequestMethod.GET)  
  17.     public BaseReturn getError() {  
  18.         int a = 0;  
  19.         int b = 1;  
  20.         LOGGER.info(b/a);  
  21.         return BaseReturn.markSuccess(null);  
  22.     }  
  23.   
  24. }  

三、执行代码观察结果

1、正常请求


  1. [INFO ] 2021-05-29 15:19:14.651 cn.hoscen.core.common.aop.TestAspect.doAround(TestAspect.java:34)  
  2. http-apr-9100-exec-4  doAround 1  
  3. [INFO ] 2021-05-29 15:19:14.651 cn.hoscen.core.common.aop.TestAspect.doBefore(TestAspect.java:29)  
  4. http-apr-9100-exec-4  doBefore  
  5. [INFO ] 2021-05-29 15:19:14.653 cn.hoscen.core.common.aop.TestAspect.doAround(TestAspect.java:36)  
  6. http-apr-9100-exec-4  doAround 2  
  7. [INFO ] 2021-05-29 15:19:14.653 cn.hoscen.core.common.aop.TestAspect.doAfter(TestAspect.java:53)  
  8. http-apr-9100-exec-4  doAfter  
  9. [INFO ] 2021-05-29 15:19:14.653 cn.hoscen.core.common.aop.TestAspect.doAfterReturning(TestAspect.java:48)  
  10. http-apr-9100-exec-4  doAfterReturning  

2、异常请求


  1. [INFO ] 2021-05-29 15:19:42.701 cn.hoscen.core.common.aop.TestAspect.doAround(TestAspect.java:34)  
  2. http-apr-9100-exec-6  doAround 1  
  3. [INFO ] 2021-05-29 15:19:42.701 cn.hoscen.core.common.aop.TestAspect.doBefore(TestAspect.java:29)  
  4. http-apr-9100-exec-6  doBefore  
  5. [INFO ] 2021-05-29 15:19:42.707 cn.hoscen.core.common.aop.TestAspect.doAfter(TestAspect.java:53)  
  6. http-apr-9100-exec-6  doAfter  
  7. [INFO ] 2021-05-29 15:19:42.707 cn.hoscen.core.common.aop.TestAspect.doAfterThrowing(TestAspect.java:42)  
  8. http-apr-9100-exec-6  doAfterThrowing  

四、得出结论与注意事项

1、执行顺序在正常和异常时有所不同,具体请看第三点的日志或自己把代码拿去跑一跑看一看。

2、对于@Around这个advice,不管它有没有返回值,但是必须要方法内部,调用一下 pjp.proceed();否则,Controller 中的接口将没有机会被执行,从而也导致了 @Before这个advice不会被触发。

3、如果同一个方法被多个Aspect类拦截呢,比如两个aspect,不管正常还是异常情况,aspect1 和 aspect2 的执行顺序都是未定的。可以通过给aspect添加@Order注解(该注解全称为:org.springframework.core.annotation.Order) 来指定顺序,值越小的 aspect 越先执行。具体情况可以下载我的测试源码自己跑一下看看就清楚了(测试同一个方法被多个Aspect类拦截,把切面类拷贝一份改个名和order就可以了),比如正常情况是下面这样

对我有用,我要     转载  
文章分类: Java  
所属标签: AspectJ  
  • 1条评论
  • 只看作者
  • 按时间|按热度
  • 小郝
    小郝   1# 2021-10-11 21:43:42.0
    赞(1) 回复 举报

    https://blog.csdn.net/hxpjava1/article/details/55504513/

  • 由于本人多次涉及需要打印这个证明,而每次都会忘记入口,在网上各种搜索各种摸索很是浪费时间。故本次将操作流程整理记录下来,以备忘。同时也分享给大家。1、打开湖北政务服务网,地址:http://zwfw.hubei.gov.cn/s/index.html2、切换区域到“武汉市”3、在“特色服务”模块找到“(个人)武汉市社会保险公共服务平台”4、进入“(个人)武汉市社会保险公共服务平台”,登录账号密码<imgsrc="https://cdnstatic.hoscen.cn/blog/article/184053017752895488/img/497065960be44747825acb86a17483c1.png"style=
  • 如何使用postman模拟http发送xml参数报文的POST请求?1、postman工具通过安装软件或使用谷歌插件都可以,这里不再赘述。2、配置postman,选择POST,填写URL;切换到Headers,添加Content-Type:text/xml 3、切换到body,选择raw,XML,下方填写你的请求报文4、点击Send发送请求,如图可以看到响应状态、时间、结果等信息5、讲到这里就结束了,是不是学会了?快去试试吧!
  • Failedtoloadprojectconfiguration:cannotparsefileF:/xx/.idea/modules.xml:ParseErrorat[row,col]:[1,1]Message:文件提前结束。解决办法:关闭idea,删掉这个文件,重新打开idea
  • 很多时候我们需要Linux服务器定时去运行一个脚本来触发一个操作,比如写缓存数据到硬盘、定时备份、定时重启服务、定期清除日志等。下面就简单讲解一下Linuxcrontab命令如何实现自动循环执行shell脚本。一、准备shell脚本比如我们准备一个hello.shvim/hcn/sh/hello.sh#!/bin/bash  DATETIME=$(date"+%Y%m%d%H%M%S") echo"hello, www.hoscen.cn,时间:${DATETIME}"  通过chmod命令赋予该脚本的执行权限chmod755hello.sh测试正确性二、开启crontab服务 linux应该都有crontab,没有的话可以安装一下:yuminstall vixie-cronyuminstall crontabsvixie-cron软
  • 目录:1、安装node.js环境2、安装cnpm3、安装vue-cli脚手架构建工具4、用vue-cli构建项目5、安装项目所需的依赖6、项目运行7、项目打包1、安装node.js环境下载地址:https://nodejs.org/zh-cn/安装过程没有太多好说的,安装完成后 win+R打开命令行输入node -v , 如图,出现版本号说明安装成功。npm包管理器是集成在node中的 , npm -v可以查看版本2、安装cnpm由于有些npm有些资源被屏蔽或者是国外资源的原因,经常会导致用npm安装依赖包的时
  • 下载地址:https://adoptopenjdk.net/releases.html?variant=openjdk8&jvmVariant=hotspot选择文件类型:或者,你可以通过我的百度网盘分享直接获取:链接:https://pan.baidu.com/s/1UygOdTh6WNZyS5WP_API6w 提取码:phnh 注意:我这里是下载的32的jdk,你们如果要64位请下载64的。使用:使用上和OracleJDK使用上是没有区别的。区别:1.OracleJDK⼤概每6个⽉发⼀次主要版本,⽽OpenJDK版本⼤概每三个⽉发布⼀次。但这不是固定的,我觉得了解这个没啥⽤处。详情参⻅:https://blogs.oracle.com/java-platform-group/update-and-faq-on-the-java-se-release-cadence。 2.OpenJDK是⼀个参考模型并且是完全开源的,⽽OracleJDK是OpenJDK的⼀个实现,并不是完全开源的; 3.OracleJDK⽐OpenJDK更稳定。OpenJDK和OracleJDK的代码⼏乎相同,但OracleJDK有更多的类和⼀些错误修复。因此,如果您
  • 建立服务器内网其他IP端口的隧道,可以将远程的服务映射到本地进行访问。finalshell配置隧道方法:
  • 本文讲触发el-dialog前动态修改窗口title的方法。1、el-dialog添加title属性el-dialog :title="titleType+'菜单'" :visible.sync="dialogVisible" width="800px" >el-dialog>  2、初始化变量(titleType,名称自己定义)export&
  • 一般我们在使用CDN时都设置有缓存时间,当源站资源发生变更后,如果缓存时间没到,那么用户访问的依旧是变更前的数据,虽说又拍云控制台提供了缓存刷新功能界面,但是每次都手动去刷新显示不太理想,当然又拍云也想到了这一点,提供给我们有API可以调用。本篇文章就是讲解如何接入又拍云缓存刷新API。网站免费接入又拍云CDN的方法,请查看我另外一篇文章,地址: https://www.hoscen.cn/blog/hao/articles/204022774975430656.html又拍云API文档:https://api.upyun.com/doc#/api/guide/overview看完文档,我们会知道又拍云提供有两个缓存刷新接口,一个支持通配符(但次数有限),一个是完整url刷新。同时注意调用接口时将 Token 放入 HTTP Header 中 。那么我们需要3个接口:1、获取token2、URL刷新3、缓存批量刷新详细请求参数和响应值请查看文档。话不多说,我们直接放上核心代码1、获取token2、URL刷新<img src="https://cdnstatic.hoscen.cn/blog/article/
  • 前言:大部分前端开发Vue用的工具是vscode 或webstorm , 但是作为后端开发习惯了使用idea, 当然开发Vue也要用idea啦。前面已经讲到了《Vue入门篇:(一)环境搭建、项目创建、项目运行、项目打包》, 接下来就讲在idea开发vue项目的环境配置。配置过程:1、打开Setting > Plugins , 搜索vue.js插件并安装。2、打开Setting >Editor >  File Types , 配置HTML文件类型支持 .vue后缀3、打开Setting > Lnaguages .. > JavaScript  , 配置ES64、打开Terminal,执行npm r