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

发布时间:2021-07-28 14:36:59 作者:小新
来源:亿速云 阅读:483

这篇文章主要介绍了如何解决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之前执行
@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;
 }

感谢你能够认真阅读完这篇文章,希望小编分享的“如何解决Spring声明式事务和@Aspect的拦截顺序的问题”这篇文章对大家有帮助,同时也希望大家多多支持亿速云,关注亿速云行业资讯频道,更多相关知识等着你来学习!

推荐阅读:
  1. MQ如何解决消息的顺序问题和消息的重复问题
  2. 如何实现spring@aspect注解aop

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

spring @aspect

上一篇:MySQL中如何使用UNIX_TIMESTAMP函数

下一篇:mysql中怎么创建存储过程

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》