在使用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
相关文章
- ASP.NET自助建站系统的域名绑定与解析教程 2025-06-10
- 个人服务器网站搭建:如何选择合适的服务器提供商? 2025-06-10
- ASP.NET自助建站系统中如何实现多语言支持? 2025-06-10
- 64M VPS建站:如何选择最适合的网站建设平台? 2025-06-10
- ASP.NET本地开发时常见的配置错误及解决方法? 2025-06-10
- 2025-07-10 怎样使用阿里云的安全工具进行服务器漏洞扫描和修复?
- 2025-07-10 怎样使用命令行工具优化Linux云服务器的Ping性能?
- 2025-07-10 怎样使用Xshell连接华为云服务器,实现高效远程管理?
- 2025-07-10 怎样利用云服务器D盘搭建稳定、高效的网站托管环境?
- 2025-07-10 怎样使用阿里云的安全组功能来增强服务器防火墙的安全性?
快网idc优惠网
QQ交流群
-
2025-05-26 81
-
2025-06-04 66
-
2025-05-29 89
-
2025-06-04 97
-
2025-05-27 39

