虚位以待(AD)
虚位以待(AD)
首页 > 软件编程 > Java编程 > Spring声明式事务和@Aspect的拦截顺序问题的解决

Spring声明式事务和@Aspect的拦截顺序问题的解决
类别:Java编程   作者:码皇   来源:互联网   点击:

本篇文章主要介绍了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之前执行@Componentpublic 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的编程式事务和声明式事务详解
  • SpringBoot 注解事务声明式事务的方式
  • 详解Spring学习之声明式事务管理
  • 完美解决Spring声明式事务不回滚的问题
  • springboot开启声明式事务的方法
  • spring声明式事务解析
  • spring声明式事务管理解析
  • SpringMVC+MyBatis声明式事务管理
相关热词搜索: Spring声明式事务 Spring @Aspect 拦截顺序