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

上一篇:Thinkphp 控制器


下一篇:关于ThinkPHP控制器的方法失效的问题