关于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. 【agc004f】Namori Grundy

    那个问一下有人可以解释以下这个做法嘛,看不太懂QwQ~ Description 有一个n个点n条边的有向图,点的编号为从1到n. 给出一个数组p,表明有(p1,1),(p2,2),…,(pn,n)这n ...

  2. js字符串排序方法

    前端开发过程中有时需自己手写排序方法 一般想到数字的字符串排序方法 我们会用到 var newArr = arr. sort(function(a,b){return a - b})来进行排序 但除此 ...

  3. ArchLinux 调节屏幕亮度和音量大小

    我使用的是dwm,所以这种琐碎小事还要我们亲自动手, 以后考虑做个脚本,把声音调节.屏幕亮度什么的统统塞进去 屏幕亮度 # 查看亮度最大值 cat /sys/class/backlight/intel ...

  4. [NOIP2012提高组]国王游戏

    题目:洛谷P1080.Vijos P1779.codevs1198. 题目大意:国王和每个大臣左.右手各写了一个数.规定每个大臣得到的金币数为他前面所有人左手的数字的乘积除以他自己右手的数(向下取整) ...

  5. NodeJS代码调试

    1.在Chrome打开chrome://flags/#enable-devtools-experiments 2.激活Developer Tools experiments 3.重启Chrome 4. ...

  6. 监控mysqld服务

    #!/bin/bash #监控mysqld服务 #telnet 192.168.122.171 3306 | grep Connected | wc -l #远程检查 #num=`netstat -n ...

  7. php后端控制可跨域的域名,允许图片跨域上传

    跨域问题经常需要面对,前端需要做的比较直接要么选择ajax异步提交,XML或者jsonp,要么表单提交前端常见跨域解决方案 jsonp基本可以搞定大部分跨域问题,但问题也比较明显,只能通过get方式提 ...

  8. 新一代企业即时通信系统 -- 傲瑞通(OrayTalk)

    傲瑞通(OrayTalk)是我们为企业专门打造的新一代企业即时通讯平台,功能强大丰富.像组织结构.文字/语音/视频会话.文件传送.远程协助.消息记录等功能都有,而且留有接口可与企业遗留系统进行集成. ...

  9. BestCoder Round #11 (Div. 2)

    太菜,仅仅能去Div2.(都做不完 ORZ... 各自是 HDU: 5054pid=5054"> Alice and Bob 5055Bob and math problem 5056 ...

  10. X的追求道路

    X的追求道路 Time Limit: 1000ms   Memory limit: 65536K  有疑问?点这里^_^ 题目描写叙述 X在大家的帮助下最终找到了一个妹纸,于是開始了漫漫的追求之路,那 ...