• 继承AbstractRoutingDataSource再通过AOP实现动态数据源切换(转)


        关于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的优先级,如下图:

  • 相关阅读:
    板邓:mysql navicat设置字段默认时间为当前时间
    板邓:wordpress用户和权限名称详细表
    板邓:jQuery设置和获取HTML、文本和值(转)
    板邓:wordpress自定义用户角色和权限全面解析
    板邓:wordpress给订阅者、投稿者上传图片权限
    七牛云
    redis 基础命令
    yeild 理解
    如何访问父类中私有的属性
    php反射
  • 原文地址:https://www.cnblogs.com/jpfss/p/8109760.html
Copyright © 2020-2023  润新知