欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2023-12-04 12:11:52
在使用abstractroutingdatasource配置多数据源时,发现使用@aspect配置的datasourceswitchaspect总是在声明式事务之后执行,配...

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

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