关于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<String, List<String>> METHODTYPE = new HashMap<String, List<String>>();
</span><span style="color: #008000">//</span><span style="color: #008000"> 获取数据源名称</span></br>
<span style="color: #0000ff">protected</span><span style="color: #000000"> Object determineCurrentLookupKey() {</br>
System.out.println(HandleDataSource.getDataSource());</br>
</span><span style="color: #0000ff">return</span><span style="color: #000000"> HandleDataSource.getDataSource();</br>
}</br> </span><span style="color: #008000">//</span><span style="color: #008000"> 设置方法名前缀对应的数据源</span></br>
<span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span> setMethodType(Map&lt;String, String&gt;<span style="color: #000000"> map) {</br>
</span><span style="color: #0000ff">for</span><span style="color: #000000"> (String key : map.keySet()) {</br>
List</span>&lt;String&gt; v = <span style="color: #0000ff">new</span> ArrayList&lt;String&gt;<span style="color: #000000">();</br>
String[] types </span>= map.get(key).split(","<span style="color: #000000">);</br>
</span><span style="color: #0000ff">for</span><span style="color: #000000"> (String type : types) {</br>
</span><span style="color: #0000ff">if</span><span style="color: #000000"> (StringUtils.isNotBlank(type)) {</br>
v.add(type);</br>
}</br>
}</br>
METHODTYPE.put(key, v);</br>
}</br>
}</br>

}

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(</span>"execution(* com.zdd.data.service.impl.*.*(..))"<span style="color: #000000">)</br>
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> aspect() {}</br> </span><span style="color: #008000">/**</span><span style="color: #008000"></br>
* 配置前置通知,使用在方法aspect()上注册的切入点</br>
</span><span style="color: #008000">*/</span><span style="color: #000000"></br>
@Before(</span>"aspect()"<span style="color: #000000">)</br>
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> before(JoinPoint point) {</br>
String className </span>=<span style="color: #000000"> point.getTarget().getClass().getName();</br>
String method </span>=<span style="color: #000000"> point.getSignature().getName();</br>
logger.info(className </span>+ "." + method + "(" + StringUtils.join(point.getArgs(), ",") + ")"<span style="color: #000000">);</br>
</span><span style="color: #0000ff">try</span><span style="color: #000000"> {</br>
L: </span><span style="color: #0000ff">for</span><span style="color: #000000"> (String key : ChooseDataSource.METHODTYPE.keySet()) {</br>
</span><span style="color: #0000ff">for</span><span style="color: #000000"> (String type : ChooseDataSource.METHODTYPE.get(key)) {</br>
</span><span style="color: #0000ff">if</span><span style="color: #000000"> (method.startsWith(type)) {
System.out.println(key);</br>
HandleDataSource.putDataSource(key);</br>
</span><span style="color: #0000ff">break</span><span style="color: #000000"> L;</br>
}</br>
}</br>
}</br>
} </span><span style="color: #0000ff">catch</span><span style="color: #000000"> (Exception e) {</br>
logger.error(e.toString());</br>
HandleDataSource.putDataSource(</span>"write"<span style="color: #000000">);</br>
}</br>
}</br> @After(</span>"aspect()"<span style="color: #000000">)</br>
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">void</span><span style="color: #000000"> after(JoinPoint point) {</br>
HandleDataSource.clear();</br>
}</br>

}




package com.zdd.data.aspect;
/**
*/


public class HandleDataSource {


// 数据源名称线程池


private static final ThreadLocal<String> holder = new ThreadLocal<String>();
</span><span style="color: #0000ff">public</span> <span style="color: #0000ff">static</span> <span style="color: #0000ff">void</span><span style="color: #000000"> putDataSource(String datasource) {</br>
holder.set(datasource);</br>
} </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">static</span><span style="color: #000000"> String getDataSource() {</br>
</span><span style="color: #0000ff">return</span><span style="color: #000000"> holder.get();</br>
}</br> </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">static</span> <span style="color: #0000ff">void</span><span style="color: #000000"> clear() {</br>
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配置文件贴上来


<bean id="dataSource" class="com.zdd.data.aspect.ChooseDataSource" lazy-init="true">

<description>数据源</description>

<property name="targetDataSources">

<map key-type="java.lang.String" value-type="javax.sql.DataSource">

<!-- write -->

<entry key="write" value-ref="writedataSource" />

<!-- read -->

<entry key="read" value-ref="readdataSource" />

</map>

</property>

<!-- 设置默认的目标数据源为读 -->

<property name="defaultTargetDataSource" ref="readdataSource" />

<property name="methodType">

<map key-type="java.lang.String">

<!-- read -->

<entry key="read" value=",get,,find,select,count,list,query," />

<!-- write -->

<entry key="write" value=",add,insert,create,update,delete,remove," />

</map>

</property>

</bean>


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

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

继承AbstractRoutingDataSource再通过AOP实现动态数据源切换(转)的更多相关文章

  1. 【原】继承AbstractRoutingDataSource再通过AOP实现动态数据源切换

    关于AbstractRoutingDataSource动态切换数据源是我在研究某开源项目时候才注意到的,大概就看懂了Spring AOP切面这里,根据作者的意思是通过继承这个抽象类可以实现数据源的动态 ...

  2. 继承AbstractRoutingDataSource再通过AOP实现动态数据源切换

    package com.zdd.data.aspect; import java.util.ArrayList; import java.util.HashMap; import java.util. ...

  3. 【开发笔记】- AbstractRoutingDataSource动态数据源切换,AOP实现动态数据源切换

    AbstractRoutingDataSource动态数据源切换 上周末,室友通宵达旦的敲代码处理他的多数据源的问题,搞的非常的紧张,也和我聊了聊天,大概的了解了他的业务的需求.一般的情况下我们都是使 ...

  4. AbstractRoutingDataSource动态数据源切换,AOP实现动态数据源切换

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明.本文链接:https://blog.csdn.net/u012881904/article/de ...

  5. Java:基于AOP的动态数据源切换(附源码)

    1 动态数据源的必要性 我们知道,物理服务机的CPU.内存.存储空间.连接数等资源都是有限的,某个时段大量连接同时执行操作,会导致数据库在处理上遇到性能瓶颈.而在复杂的互联网业务场景下,系统流量日益膨 ...

  6. Spring主从数据库的配置和动态数据源切换原理

    原文:https://www.liaoxuefeng.com/article/00151054582348974482c20f7d8431ead5bc32b30354705000 在大型应用程序中,配 ...

  7. AbstractRoutingDataSource 实现动态数据源切换原理简单分析

    AbstractRoutingDataSource 实现动态数据源切换原理简单分析 写在前面,项目中用到了动态数据源切换,记录一下其运行机制. 代码展示 下面列出一些关键代码,后续分析会用到 数据配置 ...

  8. Spring 实现动态数据源切换--转载 (AbstractRoutingDataSource)的使用

    [参考]Spring(AbstractRoutingDataSource)实现动态数据源切换--转载 [参考] 利用Spring的AbstractRoutingDataSource解决多数据源的问题 ...

  9. SpringMVC 利用AbstractRoutingDataSource实现动态数据源切换

    SpringMVC 利用AbstractRoutingDataSource实现动态数据源切换 本文转载至:http://exceptioneye.iteye.com/blog/1698064 Spri ...

随机推荐

  1. AWK行处理的用法实例

    第一节 awk的工作流程及基础用法 awk操作符会先检索文件的行信息,然后在行信息里找需要的内容. Awk的默认分割付是空格,awk '/模式/{print $1,$2}' file ##模式的位置可 ...

  2. 【Uva 11400】Lighting System Design

    [Link]: [Description] 你要构建一个供电系统; 给你n种灯泡来构建这么一个系统; 每种灯泡有4个参数 1.灯泡的工作电压 2.灯泡的所需的电源的花费(只要买一个电源就能供这种灯泡的 ...

  3. 洛谷 P3152 正整数序列

    P3152 正整数序列 题目描述 kkk制造了一个序列,这个序列里的数全是由正整数构成的.你别认为她的数列很神奇——其实就是1, 2, …, n而已.当然,n是给定的.kkk的同学lzn认为0是一个好 ...

  4. storm 消息确认机制及可靠性

    worker进程死掉 在一个节点 kill work进程 比方 kill 2509  对work没有影响 由于会在其它节点又一次启动进程运行topology任务 supervisor进程死掉 supe ...

  5. zend framework获取数据库中枚举类enum的数据并将其转换成数组

    在model中建立这种模型,在当中写入获取枚举类的方法 请勿盗版,转载请加上出处http://blog.csdn.net/yanlintao1 class Student extends Zend_D ...

  6. Linux_Oracle10 下载安装

    Linux_Oracle10 安装文档及oracle下载地址 http://download.csdn.net/detail/u012750578/8177405 一.启动数据库服务 [root@hq ...

  7. lightoj--1116--Ekka Dokka(水题)

    Ekka Dokka Time Limit: 2000MS   Memory Limit: 32768KB   64bit IO Format: %lld & %llu Submit Stat ...

  8. export和source的区别

    1.执行脚本是在一个子shell环境运行的,脚本执行完后该子shell自动退出. 2.执行脚本中的系统环境变量(用export定义的变量)才会被复制到子shell中. 3.一个shell中的系统环境变 ...

  9. 如何使用通用pe工具箱破解开机密码

    下载最新版的通用pe工具箱将u盘制作成启动盘,接着重启连续按热键进入到bios系统下,设置u盘为第一启动,保存重启. 1.这时候会进入通用pe工具箱的选择界面,我们选择第八个“运行Windows登陆密 ...

  10. 无闻go编程基础笔记

    Go语言做Web编程非常方便,并且在开发效率和程序运行效率方面都非常优秀.相比于Java,其最大的优势就是简便易用,而相比于PHP,它最大的优势就是性能好. (go做web)推荐Gorilla的库,里 ...