Spring声明式事务和@Aspect的拦截顺序问题的解决

网友投稿 188 2023-08-06

Spring声明式事务和@Aspect的拦截顺序问题的解决

在使用AbstractRoutingDataSource配置多数据源时,发现使用@aspect配置的DataSourceSwitchAspect总是在声明式事务之后执行,配置了Order依然不行,经过调研发现是由于两者的aop代理方式不一致导致。

在spring内部,是通过BeanPostProcessor(《spring 攻略》一书中翻译为,后处理器)来完成自动创建代理工作的。根据匹配规则的不同大致分为三种类别: 1、匹配Bean的名称自动创建匹配到的Bean的代理,实现类BeanNameAutoProxyCreator 2、根据Bean中的AspectJ注解自动创建代理,实现类AnnotationAwareAspectJAutoProxyCreator 3、根据Advisor的匹配机制自动创建代理,会对容器中所有的Advisor进行扫描,自动将这些切面应用到匹配的Bean中,实现类DefaultAdvisorAutoProxyCreator

其中@Aspect声明的aop是通过AnnotationAwareAspectJAutoProxyCreator进行代理的,而项目中的声明式事务是BeanNameAutoProxyCreator方式进行代理的,经调试发现BeanNameAutoProxyCreator拦截优先级高于AnnotationAwareAspectJAutoProxyCreator,order配置只对同一类型的aop拦截方式起作用,如下:

DataSourceSwitchAspect

/**

* 数据源切换切面

* @author Matchstick

*/

@Aspect

@Order(1) //确保该切面在transaction之前执行

@Component

public class DataSourceSwitchAspect

{

private Logger logger = LoggerFactory.getLogger(getClass());

@Pointcut("@annotation(com.etu.multidatasource.test.datasource.DataSourceId)")

public void pointcut(){}

@Before("@annotation(dataSourceId)")

public void switchDataSource(JoinPoint point, DataSourceId dataSourceId)

{

String dsId = dataSourceId.value();

MultiDataSourceContextHolder.setDataSourceId(dsId);

logger.debug("switch datasource -> {}", dsId);

}

@After("@annotation(dataSourceId)")

public void restoreDataSource(JoinPoint point, DataSourceId dataSourceId)

{

MultiDataSourceContextHolder.removeDataSourceId();

logger.debug("restore datasource -> {}", MultiDataSourceContextHolder.getDefaultDataSourceId());

}

}

DataSourceConfig

@Bean

public BeanNameAutoProxyCreator txProxy()

{

BeanNameAutoProxyCreator creator = new BeanNameAutoProxyCreator();

creator.setInterceptorNames("txAdvice");

creator.setBeanNames("*Service", "*ServiceImpl");

creator.setProxyTargetClass(true);

creator.setOrder(2);

return creator;

}

解决方案:要么修改DataSourceSwitchAspect的aop方式为BeanNameAutoProxyCreator,要么修改事务aop方式为AnnotationAwareAspectJAutoProxyCreator,由于是通过注解实现的数据源切换aop,所以选择了后者解决方案,如下:

DataSourceConfig

@Bean

public AnnotationAwareAspectJAutoProxyCreator txProxy()

{

/*

* 必须使用AsCqcUMpectJ方式的AutoProxy,这样才能和DataSourceSwitchAspect保持统一的aop拦截方式,否则不同的拦截方式会导致order失效

*/

AnnotationAwareAspectJAutoProxyCreator c = new AnnotationAwareAspectJAutoProxyCreator();

c.setInterceptorNames("txAdvice");

c.setIncludePatterns(Arrays.asList("execution (public com.etu..*Service(..))"));

c.setProxyTargetClass(true);

c.setOrder(2);

return c;

}

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:浅谈Angular HttpClient简单入门
下一篇:Java实现终止线程池中正在运行的定时任务
相关文章

 发表评论

暂时没有评论,来抢沙发吧~