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 aop无法拦截代理内部函数调用的问题
- 4000 spring解决hibernate懒加载的问题
- spring boot log4j2与三方依赖库log4j冲突无法初始化问题解决方法
- Spring Cloud Edgware新特性之一:解决Eureka中Jersey 1.x版本过旧的问题-不使用Jersey
- Spring学习总结(20)——Spring加载多个项目properties配置文件问题解决
- 解决spring-mvc @responseBody注解返回json 乱码问题
- 在Spring整合 SpringMVC,SpringData和 JPA 时,如何解决 Lazy懒加载问题?
- struts2,hibernate4,spring3配置时问题汇总及解决办法
- 解决spring boot websocket无法注入bean的问题
- java.lang.NoSuchMethodError: org.springframework.beans.factory.xml...setEnviro问题解决方法
- SWF和远程链接交互,本地直接运行SWF格式的文件会出现拦截问题的解决
- SpringDataJpa -- NoSession问题分析和解决
- Spring 报错:元素 "context:component-scan" 的前缀 "context" 未绑定的问题解决
- 解决spring-boot启动中碰到的问题:Cannot determine embedded database driver class for database type NONE
- 解决MAVEN用assembly打包spring.handlers和spring.schemas出错的问题
- 彻底解决SEP11弹出拦截ntoskrnl.exe通讯问题
- Spring import配置文件的顺序问题
- 解决Spring中try catch 无法回滚问题
- 让spring帮助你在MVC层解决JPA的缓迟加载问题
- 让spring帮助你在MVC层解决JPA的缓迟加载问题