AOP切面不生效场景
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了AOP切面不生效场景相关的知识,希望对你有一定的参考价值。
参考技术A 常用AOP不生效场景:1、未配置AOP生效
Springboot:
@EnableAspectJAutoProxy(proxyTargetClass=true, exposeProxy=true)
Spring xml配置:
<aop:aspectj-autoproxy proxy-target-class="true" />
2、不能使用AOP切面方法调用本类的方法
例如SampleServiceA存在两个方法A和B,现对SampleServiceA中的方法B做切面,
场景1:SampleServiceB调用SampleServiceA的方法B,则切面生效;
场景2:某个service调用SampleServiceA的A方法,然后A方法内部调用了B方法,此时切面不生效;
3、切面表达式写错,百度之
Spring 嵌套方法AOP不生效问题
问题描述, 如下Abc定义为一个Bean, b()方法添加@TargetDatasource,定义切面DynamicDataSourceAspect,期望:调用a()方法,b()方法上的AOP拦截能生效。实际不生效。
@Service public class Abc { public void a(){
b();
}
@TargetDatasource(value=abc) public void b(){
} }
AOP代码:
@Order(-1) @Component @Slf4j @Aspect public class DynamicDataSourceAspect { /** * set data source value. */ @Before(value = "@annotation(targetDataSource)", argNames = "targetDataSource") public void setDataSource(TargetDataSource targetDataSource) { if (targetDataSource != null && targetDataSource.value() != null) { log.info("Databse value: {}", targetDataSource.value()); DynamicDataSourceHolder.set(targetDataSource.value()); } } /** * clear data source value. */ @After(value = "@annotation(targetDataSource)", argNames = "targetDataSource") public void clearDataSource(TargetDataSource targetDataSource) { DynamicDataSourceHolder.clear(); } }
问题分析:
我们都知道Spring aop有两种实现方式,基于Interface生成代理和cglib生成代理。上面的例子中,当调用a()方法时,调用者拿到的是Abc的代理类,即增强类。如果a()方法上有@TargetDatasource注解,拦截会生效。然而,在a()方法里调用b(),b方法上的拦截不会生效。原因是因为a()调用b(), 用的是b()方法的目标类,而不是代理类,所以拦截不生效。
解决方法:
1. 重构代码, 把b()方法移到一个Bean里面。
2. 调整a()方法如下:
@Service public class Abc { public void a(){ ((Abc)AopContext.currentProxy()).b(); } @TargetDatasource(value=abc) public void b(){ } }
Sping 官方解释 - https://docs.spring.io/spring/docs/current/spring-framework-reference/core.html#aop
以上是关于AOP切面不生效场景的主要内容,如果未能解决你的问题,请参考以下文章