Spring声明式事务和@Aspect的拦截顺序问题的解决
作者:Matchstick 发布时间:2023-07-18 12:10:46
在使用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;
}
来源:https://my.oschina.net/u/2333620/blog/1805869


猜你喜欢
- public class TestSqlserverJtds { public static void main(String[]
- 单例模式创建唯一的一个变量(对象),在类中将构造函数设为protected或者private(析构函数设为相对应的访问权限),故外部不能实例
- SpringBoot访问外部文件及默认路由1 新增配置类package com.pibigstar.common.config;import
- 本文实例讲述了Android实现将应用崩溃信息发送给开发者并重启应用的方法。分享给大家供大家参考,具体如下:在开发过程中,虽然经过测试,但在
- 我想,对于各位使用面向对象编程语言的程序员来说,“接口”这个名词一定不陌生,但是不知各位有没有这样的疑惑:接口有什么用途?它和抽象类有什么区
- 1. reference:参考某一资源ID。(1)属性定义:<declare-styleable name = "名称&qu
- 本文实例讲述了Android开发实现popupWindow弹出窗口自定义布局与位置控制方法。分享给大家供大家参考,具体如下:布局文件:主布局
- Java序列化JSON时long型数值,会出现精度丢失的问题。原因:java中得long能表示的范围比js中number大,也就意味着部分数
- 对这种懒加载问题,最后的做法是利用Spring提供的一个针对Hibernate的一个支持类,其主要意思是在发起一个页面请求时打开Hibern
- 数据库事务是被当作单个工作单元的操作序列。这些操作要么全部完成或全部不成功。事务管理是面向企业应用程序,以确保数据的完整性和一致性RDBMS
- 前言内存治理一直是每个开发者最关心的问题,我们在日常开发中会遇到各种各样的内存问题,比如OOM,内存泄露,内存抖动等等,这些问题都有以下共性
- SSL是为网络通信提供安全以及保证数据完整性的的一种安全协议,SSL在网络传输层对网络连接进行加密。例:cas 的单点登陆就用到了SSL一、
- 最近项目需求中有需要导出Execl表格的要求,而且还是大量的数据,于是把之前的整理了一下,抽成了一个统一的工具类,需要时直接调用工具类即可,
- static目的java中的static关键字主要用于内存管理。static范围使用范围:java static关键字可以用在变量、方法、代
- 1.使用的注意事项本节给大家带来基础UI控件部分的最后一个控件:DrawerLayout,官方给我们提供的一个侧滑菜单控件,和上一节的Vie
- 1、需求及配置需求:爬取京东手机搜索页面的信息,记录各手机的名称,价格,评论数等,形成一个可用于实际分析的数据表格。使用Maven项目,lo
- 在C#中数组Array,ArrayList,泛型List都能够存储一组对象,但是在开发中根本不知道用哪个性能最高,下面我们慢慢分析分析。一、
- 如下所示:package cn.jdk.foreach;import java.util.HashMap;import java.util.
- 一、什么是JWTJSON Web Token (JWT),它是目前最流行的跨域身份验证解决方案。现在的项目开发一般都是前端端分离,这就涉及到
- 一、查看官网http://www.zhuozhengsoft.com/点击首页下载,进入页面:最新得5.2,我们就下载5.2版本进行测试。二