时间:2021-05-02
在使用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
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 /** * 数据源切换切面 * @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
? 1 2 3 4 5 6 7 8 9 10 @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
? 1 2 3 4 5 6 7 8 9 10 11 12 13 @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; }以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://my.oschina.net/u/2333620/blog/1805869
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
前言spring事务管理包含两种情况,编程式事务、声明式事务。而声明式事务又包括基于注解@Transactional和tx+aop的方式。那么本文先分析编程式注
前沿:通过对spring事务管理有了比较深入学习,本文将不做实例,而是指定具体的类和配置文件进行讲解。本文内容: 1、了解什么是声明式事务? 2、声明式事务
编程式事务在Spring中事务管理的方式有两种,编程式事务和声明式事务。先详细介绍一下两种事务的实现方式.配置类@Configuration@EnableTra
一、编程式事务二、声明式事务1、基于XML的事务1.1Spring配置文件1.2业务类和下面注解方式使用的类一样,不过是去掉了注解,我将所有的Service层放
本文实例讲述了Spring实战之使用TransactionProxyFactoryBean实现声明式事务操作。分享给大家供大家参考,具体如下:一配置文件