Spring AOP方法内部调用不生效的解决方案
问题描述
最近有个需求,统计某个方法的调用次数,开始使用 Spring AOP 实现,后来发现当方法被内部调用时,切面逻辑将不会生效,直接上样例:
定义接口,包含方法 A,方法 B
public interface ISon { void A(); void B(); }
定义接口实现,方法 B 调用方法 A
@Service public class Son implements ISon { @Override public void A() { System.out.println("method A"); } @Override public void B() { System.out.println("method B"); A(); } }
切点定义,对方法 A 进行增强
@Aspect @Component public class AspectDemo { @Pointcut(value = "execution(* com.example.transactiondemo.aop.ISon.A(..))") public void pointCut(){ } @Before("pointCut()") public void before(JoinPoint joinPoint){ System.out.println("before "); } @After("pointCut()") public void after(){ System.out.println("after "); } }
测试
@Autowired private ISon iSon; @Test public void test11(){ iSon.A(); System.out.println("-----------"); iSon.B(); }
打印输出
before
method A
after
-----------
method B
method A
可以发现,如果直接使用 实例调用 A,会进行增强;但是如果是方法 B 调用,A 的增强逻辑将失效。
失效原因
方法 A 被调用,是基于 AOP 生成的 代理对象 进行的调用;方法 B 调用方法 A ,是 this 目标对象 直接调用,并不是代理对象进行调用
解决方案
回到最开始的需求,如何通过切面的方式获取某一个方法的调用次数? 包括方法自调用
使用 AopContext.currentProxy()
切面开启注解
@EnableAspectJAutoProxy(exposeProxy = true,proxyTargetClass = true)
方法 B 不直接调用 A
@Override public void B() { System.out.println("method B"); ((Son) AopContext.currentProxy()).A(); }
先获取上下文的 Bean 再调用
@Autowired private ApplicationContext applicationContext; ((Son) applicationContext.getBean(Son.class)).A();
以上就是Spring AOP方法内部调用不生效的解决方案的详细内容,更多关于Spring AOP方法调用不生效的资料请关注脚本之家其它相关文章!
相关文章
解决Mybatis-plus自定义TypeHandler查询映射结果一直为null问题
这篇文章主要介绍了解决Mybatis-plus自定义TypeHandler查询映射结果一直为null问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教2024-07-07java客户端Jedis操作Redis Sentinel 连接池的实现方法
下面小编就为大家带来一篇java客户端Jedis操作Redis Sentinel 连接池的实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧2017-03-03解决mybatis generator MySQL自增ID出现重复问题MySQLIntegrityC
在MySQL中使用MyBatis时,可能会遇到由于主键重复导致的插入失败问题,此问题通常发生在连续插入多条数据时,如果selectKey的order配置错误,如使用BEFORE而不是AFTER,将会导致获取的ID未更新,引起主键重复错误,正确的配置应使用AFTER2024-10-10
最新评论