Spring聲明式事務和@Aspect的攔截順序問題的解決
在使用AbstractRoutingDataSource配置多數(shù)據(jù)源時,發(fā)現(xiàn)使用@aspect配置的DataSourceSwitchAspect總是在聲明式事務之后執(zhí)行,配置了Order依然不行,經過調研發(fā)現(xiàn)是由于兩者的aop代理方式不一致導致。
在spring內部,是通過BeanPostProcessor(《spring 攻略》一書中翻譯為,后處理器)來完成自動創(chuàng)建代理工作的。根據(jù)匹配規(guī)則的不同大致分為三種類別: 1、匹配Bean的名稱自動創(chuàng)建匹配到的Bean的代理,實現(xiàn)類BeanNameAutoProxyCreator 2、根據(jù)Bean中的AspectJ注解自動創(chuàng)建代理,實現(xiàn)類AnnotationAwareAspectJAutoProxyCreator 3、根據(jù)Advisor的匹配機制自動創(chuàng)建代理,會對容器中所有的Advisor進行掃描,自動將這些切面應用到匹配的Bean中,實現(xiàn)類DefaultAdvisorAutoProxyCreator
其中@Aspect聲明的aop是通過AnnotationAwareAspectJAutoProxyCreator進行代理的,而項目中的聲明式事務是BeanNameAutoProxyCreator方式進行代理的,經調試發(fā)現(xiàn)BeanNameAutoProxyCreator攔截優(yōu)先級高于AnnotationAwareAspectJAutoProxyCreator,order配置只對同一類型的aop攔截方式起作用,如下:
DataSourceSwitchAspect
/** * 數(shù)據(jù)源切換切面 * @author Matchstick */ @Aspect @Order(1) //確保該切面在transaction之前執(zhí)行 @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,由于是通過注解實現(xiàn)的數(shù)據(jù)源切換aop,所以選擇了后者解決方案,如下:
DataSourceConfig
@Bean public AnnotationAwareAspectJAutoProxyCreator txProxy() { /* * 必須使用AspectJ方式的AutoProxy,這樣才能和DataSourceSwitchAspect保持統(tǒng)一的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; }
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
SpringBoot 如何根據(jù)不同profile選擇不同配置
這篇文章主要介紹了SpringBoot 如何根據(jù)不同profile選擇不同配置的操作方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08Spring?Initializr只能創(chuàng)建為Java?17版本以上的問題解決
這篇文章主要給大家介紹了關于Spring?Initializr只能創(chuàng)建為Java?17版本以上問題的解決辦法,文中通過圖文介紹的非常詳細,對大家的學習或者工作具有一定的參考借鑒價值,需要的朋友可以參考下2024-01-01springboot如何查找配置文件路徑的順序和其優(yōu)先級別
此文是在工作中遇到的關于springboot配置文件的問題,在網上查閱資料和自己測試之后記錄的,以便日后查閱。希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08