Springboot内部方法调用,aop切面失效
场景描述
有一个控制层类OutStoreOverController(简称controller),依赖了XsCustomorExpenseOperateServiceImpl(简称service)类。controller在2个不同方法中分别调用了service的siteDeliverySettlement和stockDownAccounts方法(分别简称为m1和m2)。m1和m2在具体实现的时候又调用了service的内部方法createExpense(申明为public,简称为m3)
方法调用的时序图如下:
现在有一个切面StorageOperateOMSAopServiceImpl,需要切createExpense(m3)方法,在m3方法执行前做点事情。经过配置后,运行发现m3方法并没有被切到
问题分析
当controller构建实例的时候,注入service实例的时候,发现其有切面,产生了代理类serviceProxy并注入给了controller
实际调用的时序图如下
这样就导致m3方法根本没有被切面切入。虽然controller第一次调用的是代理类,但是在调用m3方法的时候是调用的service实例内部的m3方法,所以切面没有生效。
解决问题
1 XXXXXXXXXX; 2 m3(); 3 XXXXXXXX; 4 修改后的写法为: 5 XXXXXXXX; 6 Service serviceTemp=ApplicationContextUtil.getBean(Service.class); 7 serviceTemp.m3(); 8 XXXXXX;
修改后调用的时序图为
总结
真正使切面生效的就是:Service serviceTemp=ApplicationContextUtil.getBean(Service.class); 这一行代码。向spring容器拿的实例,实际上是代理类servciceProxy。调用代理类的m3方法就会去先执行aop中前置切面代码,再会调用真正service实例的m3方法。最终,aop才有效果了。需要理解基于动态代理的aop原理。
也可以使用注解在当前类中注入本身,详见另一篇博客。
参考:
https://blog.csdn.net/jishanwang/article/details/86748328
相关文章
- SpringBoot单元测试
- SpringBoot注解把配置文件自动映射到属性和实体类实战
- SpringBoot中Junit测试注入Bean失败的解决方法
- SpringBoot 如何进行对象复制
- 【SpringBoot笔记16】SpringBoot集成Quartz实现定时任务功能
- Springboot项目快速实现过滤器功能
- 【SpringBoot系列】最详细demo- 集成MyBatis
- 【SpringBoot系列】最详细demo-- 怎么不用定时任务实现关闭订单
- 基于注解SpringAOP,AfterReturning,Before,Around__springboot工程 @Around 简单的使用__SpringBoot:AOP 自定义注解实现日志管理
- Springboot如何实现自动输出word文档功能?实例代码解析(word文档答题试卷)
- SpringBoot项目中Controller层代码编写规范整理
- SpringBoot升级到2.0后默认时间格式变化_springboot接收date类型参数
- springboot配置跨mapper.xml的全局变量
- SpringBoot的 CommandLineRunner的使用
- SpringBoot实例②springboot+jdbcTemplate小例子实现增删查改
- 基于Java+SpringBoot+Vue前后端分离即时通讯系统设计与实现
- springboot使用redis(StringRedisTemplate的常用方法)
- SpringBoot 激活 profiles