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

2025-05-29 0 78

在使用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;

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持快网idc。

原文链接:https://my.oschina.net/u/2333620/blog/1805869

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 Spring声明式事务和@Aspect的拦截顺序问题的解决 https://www.kuaiidc.com/112029.html

相关文章

发表评论
暂无评论