博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
继承AbstractRoutingDataSource再通过AOP实现动态数据源切换(转)
阅读量:6470 次
发布时间:2019-06-23

本文共 4632 字,大约阅读时间需要 15 分钟。

    关于AbstractRoutingDataSource我在研究开源中国的开源项目时候才发现,好奇的看了一下代码发现自己看不明白,大概就看懂了Spring AOP切面这里,根据注释作者的意思是通过这个可以实现数据源的动态切换,也就是Controller调用Service的时候会切换数据源。最终研究了一天也没发现什么结果,第二天便尝试查看源码和查看相关资料,试图揭开这个疑惑


  •    首先贴上源代码如下:
package com.zdd.data.aspect; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import org.apache.commons.lang3.StringUtils; import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; /**  * 获取数据源 *  */ public class ChooseDataSource extends AbstractRoutingDataSource {
public static Map
> METHODTYPE = new HashMap
>(); // 获取数据源名称 protected Object determineCurrentLookupKey() {
System.out.println(HandleDataSource.getDataSource()); return HandleDataSource.getDataSource(); } // 设置方法名前缀对应的数据源 public void setMethodType(Map
map) {
for (String key : map.keySet()) {
List
v = new ArrayList
(); String[] types = map.get(key).split(","); for (String type : types) {
if (StringUtils.isNotBlank(type)) {
v.add(type); } } METHODTYPE.put(key, v); } } }
package com.zdd.data.aspect; import org.apache.commons.lang3.StringUtils; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.After; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.slf4j.LoggerFactory; import org.springframework.context.annotation.EnableAspectJAutoProxy; import org.springframework.core.annotation.Order; import org.springframework.stereotype.Component; @Aspect @EnableAspectJAutoProxy(proxyTargetClass = true) @Component public class DataSourceAspect {
private final org.slf4j.Logger logger = LoggerFactory.getLogger(DataSourceAspect.class); @Pointcut("execution(* com.zdd.data.service.impl.*.*(..))") public void aspect() {} /** * 配置前置通知,使用在方法aspect()上注册的切入点 */ @Before("aspect()") public void before(JoinPoint point) {
String className = point.getTarget().getClass().getName(); String method = point.getSignature().getName(); logger.info(className + "." + method + "(" + StringUtils.join(point.getArgs(), ",") + ")"); try {
L: for (String key : ChooseDataSource.METHODTYPE.keySet()) {
for (String type : ChooseDataSource.METHODTYPE.get(key)) {
if (method.startsWith(type)) { System.out.println(key); HandleDataSource.putDataSource(key); break L; } } } } catch (Exception e) {
logger.error(e.toString()); HandleDataSource.putDataSource("write"); } } @After("aspect()") public void after(JoinPoint point) {
HandleDataSource.clear(); } }
package com.zdd.data.aspect; /** */ public class HandleDataSource {
// 数据源名称线程池 private static final ThreadLocal
holder = new ThreadLocal
(); public static void putDataSource(String datasource) {
holder.set(datasource); } public static String getDataSource() {
return holder.get(); } public static void clear() {
holder.remove(); }}

  以上是核心代码,我疑惑的地方在于AbstractRoutingDataSource这里,刚开始去看的时候不明白为什么继承这个就能实现数据源切换,最后进入AbstractRoutingDataSource.class去查看究竟,发现继承了一个AbstractDataSource的抽象类,这里我突然有点明白AbstractRoutingDataSource的意思,一个路由数据源,一般用到Routing的地方都可以进行动态配置。

  • 大致翻译一下源码作者对这个类的说明:调用一个基于各种目标数据源

      继续往下看又看到几个重要信息

   

     上图发现2个方法分别是setTargetDataSources(注入目标数据源) setDefaultTargetDataSource(注入默认的数据源);然后脑海里想到是可以通过spring xml 配置来注入这2个方法,带着这个思绪继续接着继续往下看;关于看源码一定要看个究竟,不然云里雾绕的。然后又看到比较重要的信息如下:

 上图发现有getConnection方法,jdbc获取连接的方法,而当我点进determineTargetDataSource方法时得到了一个确切答案,代码如下:

   上图大致的意思是会调用determineCurrentLookupKey这个抽象方法获取数据源,如果是null则获取默认的数据源,反之则是获取我们注入的数据源, 这样就实现了数据源的动态切换, 这时候我通过断点启动也验证了这一个流程确实如此。

 


 

   分析了一个动态切换流程后,那么问题来了,如何让我们的代码实现这个功能呢?其实就是重写AbstractRoutingDataSource方法后再通过aop动态切换数据源,那么如何知道切换条件是什么?这时候可以根据Service层的方法来确定,一般我们Service都是会以find,add,delete,update 开头; 例如发现是find,get开头则走读库,delete,update则走写库。

 

  分析完后把spring配置文件贴上来

数据源

   上面的配置大概流程是先指定2个数据源注入targetDataSources,然后配置一组数组用来存放判断走读库还是写库的条件,如果是read那么肯定是把read注入的数据源拿出来,如果是write则把write的数据源取出来。

 

   但是这时候会有一个奇怪的现象就是每次调用的时候会走chooseDataSource再走AOP,方法解决很简单,设置一下AOP的优先级,如下图:

 

转载地址:http://mscko.baihongyu.com/

你可能感兴趣的文章
Android 自动化测试—robotium(二)初识
查看>>
VMware-桌面虚拟化(二)
查看>>
haproxy 重启提示:cannot bind socket(无法绑定socket)
查看>>
与RMAN有关的初始化参数和动态性能视图
查看>>
2017下半年计划
查看>>
nginx+keepalived做高可用
查看>>
安装盘启动系统,挂载硬盘,然后进入shell模式修改配置文件
查看>>
安装 Active Directory 架构管理单元
查看>>
Linux系统巡检常用命令
查看>>
C++ 中字符串与数值的相互转换
查看>>
邮件江湖群狼环伺 U-Mail邮件系统防狼有术
查看>>
全球五大顶级域名统计:5月第三周新增20.3万个
查看>>
MFS--分布式文件系统
查看>>
我的友情链接
查看>>
nagios下 监控内存的插件
查看>>
linux磁盘分区
查看>>
传说FreeBSD等比Linux更稳定,更“健壮”
查看>>
cmake安装MySQL数据库实例
查看>>
朝韩合并的几个好处
查看>>
<05>linux的文本基础操作
查看>>