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()
{
/*
* 必须使用AspectJ方式的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;
}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
相关文章
Spring Security系列教程之会话管理处理会话过期问题
会话过期,是指当用户登录网站后,较长一段时间没有与服务器进行交互,将会导致服务器上的用户会话数据(即session)被销毁。这篇文章主要介绍了Spring Security系列教程之会话管理处理会话过期问题,需要的朋友可以参考下2021-10-10
springboot实现SSE(Server Sent Event)的示例代码
SSE 全称Server Sent Event,直译一下就是服务器发送事件,本文主要为大家详细介绍了springboot实现SSE的相关知识,需要的可以参考一下2024-04-04
关于swagger配置及踩坑@Api参数postion无效解决接口排序问题
这篇文章主要介绍了关于swagger配置及踩坑@Api参数postion无效解决接口排序问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教2022-06-06


最新评论