Spring 嵌套方法AOP不生效问题

Posted 码农在路上

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了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 






以上是关于Spring 嵌套方法AOP不生效问题的主要内容,如果未能解决你的问题,请参考以下文章

为啥在aop里面的事务不生效

SpringMVC中配aop拦截不生效,咋回事

AOP切面不生效场景

揭秘 Spring AOP 失效的罪因!

SpringMVC中配aop拦截不生效,咋回事

Spring 事务失效的7种场景