一种实现Spring动态数据源切换的方法

打印 上一主题 下一主题

主题 838|帖子 838|积分 2514

1 目标

不在现有查询代码逻辑上做任何改动,实现dao维度的数据源切换(即表维度)
2 使用场景

节约bdp的集群资源。接入新的宽表时,通常uat验证后就会停止集群释放资源,在对应的查询服务器uat环境时需要查询的是生产库的表数据(uat库表因为bdp实时任务停止,没有数据落入),只进行服务器配置文件的改动而无需进行代码的修改变更,即可按需切换查询的数据源。
2.1 实时任务对应的集群资源


2.2 实时任务产生的数据进行存储的两套环境


2.3 数据使用系统的两套环境(查询展示数据)


即需要在zhongyouex-bigdata-uat中查询生产库的数据。
3 实现过程

3.1 实现重点


  • org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource
    spring提供的这个类是本次实现的核心,能够让我们实现运行时多数据源的动态切换,但是数据源是需要事先配置好的,无法动态的增加数据源。
  • Spring提供的Aop拦截执行的mapper,进行切换判断并进行切换。
注:另外还有一个就是ThreadLocal类,用于保存每个线程正在使用的数据源。
3.2 AbstractRoutingDataSource解析
  1. public abstract class AbstractRoutingDataSource extends AbstractDataSource
  2. implements InitializingBean{
  3.     @Nullable
  4.     private Map<Object, Object> targetDataSources;
  5.     @Nullable
  6.     private Object defaultTargetDataSource;
  7.     @Override
  8.     public Connection getConnection() throws SQLException {
  9.         return determineTargetDataSource().getConnection();
  10.     }
  11.     protected DataSource determineTargetDataSource() {
  12.         Assert.notNull(this.resolvedDataSources, "DataSource router not initialized");
  13.         Object lookupKey = determineCurrentLookupKey();
  14.         DataSource dataSource = this.resolvedDataSources.get(lookupKey);
  15.         if (dataSource == null && (this.lenientFallback || lookupKey == null)) {
  16.             dataSource = this.resolvedDefaultDataSource;
  17.         }
  18.         if (dataSource == null) {
  19.             throw new IllegalStateException("Cannot determine target DataSource for lookup key [" + lookupKey + "]");
  20.         }
  21.         return dataSource;
  22.     }
  23.     @Override
  24.     public void afterPropertiesSet() {
  25.         if (this.targetDataSources == null) {
  26.             throw new IllegalArgumentException("Property 'targetDataSources' is required");
  27.         }
  28.         this.resolvedDataSources = new HashMap<>(this.targetDataSources.size());
  29.         this.targetDataSources.forEach((key, value) -> {
  30.             Object lookupKey = resolveSpecifiedLookupKey(key);
  31.             DataSource dataSource = resolveSpecifiedDataSource(value);
  32.             this.resolvedDataSources.put(lookupKey, dataSource);
  33.         });
  34.         if (this.defaultTargetDataSource != null) {
  35.             this.resolvedDefaultDataSource = resolveSpecifiedDataSource(this.defaultTargetDataSource);
  36.         }
  37.     }
复制代码
从上面源码可以看出它继承了AbstractDataSource,而AbstractDataSource是javax.sql.DataSource的实现类,拥有getConnection()方法。获取连接的getConnection()方法中,重点是determineTargetDataSource()方法,它的返回值就是你所要用的数据源dataSource的key值,有了这个key值,resolvedDataSource(这是个map,由配置文件中设置好后存入targetDataSources的,通过targetDataSources遍历存入该map)就从中取出对应的DataSource,如果找不到,就用配置默认的数据源。
看完源码,我们可以知道,只要扩展AbstractRoutingDataSource类,并重写其中的determineCurrentLookupKey()方法返回自己想要的key值,就可以实现指定数据源的切换!
3.3 运行流程


  • 我们自己写的Aop拦截Mapper
  • 判断当前执行的sql所属的命名空间,然后使用命名空间作为key读取系统配置文件获取当前mapper是否需要切换数据源
  • 线程再从全局静态的HashMap中取出当前要用的数据源
  • 返回对应数据源的connection去做相应的数据库操作
3.4 不切换数据源时的正常配置
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
  5.     <bean id="dataSourceClickhousePinpin"  destroy-method="close" lazy-init="true">
  6.         <property name="url" value="${clickhouse.jdbc.pinpin.url}" />
  7.     </bean>
  8.     <bean id="singleSessionFactoryPinpin" >
  9.         
  10. <property name="dataSource" ref="dataSourceClickhousePinpin" />
  11.     </bean>
  12. </beans>
复制代码
3.5 进行动态数据源切换时的配置
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
  5.     <bean id="dataSourceClickhousePinpin"  destroy-method="close" lazy-init="true">
  6.         <property name="url" value="${clickhouse.jdbc.pinpin.url}" />
  7.     </bean>
  8.     <bean id="singleSessionFactoryPinpin" >
  9.         
  10. <property name="dataSource" ref="dataSourceClickhousePinpin" />
  11.     </bean>
  12. </beans><?xml version="1.0" encoding="UTF-8"?>
  13. <beans xmlns="http://www.springframework.org/schema/beans"
  14.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
  15.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
  16.     <bean id="dataSourceClickhousePinpin"  destroy-method="close" lazy-init="true">
  17.         <property name="url" value="${clickhouse.jdbc.pinpin.url}" />
  18.     </bean>
  19.     <bean id="singleSessionFactoryPinpin" >
  20.         
  21. <property name="dataSource" ref="dataSourceClickhousePinpin" />
  22.     </bean>
  23. </beans><?xml version="1.0" encoding="UTF-8"?>
  24. <beans xmlns="http://www.springframework.org/schema/beans"
  25.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
  26.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
  27.     <bean id="dataSourceClickhousePinpin"  destroy-method="close" lazy-init="true">
  28.         <property name="url" value="${clickhouse.jdbc.pinpin.url}" />
  29.     </bean>
  30.     <bean id="singleSessionFactoryPinpin" >
  31.         
  32. <property name="dataSource" ref="dataSourceClickhousePinpin" />
  33.     </bean>
  34. </beans><?xml version="1.0" encoding="UTF-8"?>
  35. <beans xmlns="http://www.springframework.org/schema/beans"
  36.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
  37.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
  38.     <bean id="dataSourceClickhousePinpin"  destroy-method="close" lazy-init="true">
  39.         <property name="url" value="${clickhouse.jdbc.pinpin.url}" />
  40.     </bean>
  41.     <bean id="singleSessionFactoryPinpin" >
  42.         
  43. <property name="dataSource" ref="dataSourceClickhousePinpin" />
  44.     </bean>
  45. </beans>   
复制代码
核心是AbstractRoutingDataSource,由spring提供,用来动态切换数据源。我们需要继承它,来进行操作。这里我们自定义的com.zhongyouex.bigdata.common.aop.MultiDataSource就是继承了AbstractRoutingDataSource
  1. package com.zhongyouex.bigdata.common.aop;
  2. import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
  3. /**
  4. * @author: cuizihua
  5. * @description: 动态数据源
  6. * @date: 2021/9/7 20:24
  7. * @return
  8. */
  9. public class MultiDataSource extends AbstractRoutingDataSource {
  10.     /* 存储数据源的key值,InheritableThreadLocal用来保证父子线程都能拿到值。
  11.      */
  12.     private static final ThreadLocal<String> dataSourceKey = new InheritableThreadLocal<String>();
  13.     /**
  14.      * 设置dataSourceKey的值
  15.      *
  16.      * @param dataSource
  17.      */
  18.     public static void setDataSourceKey(String dataSource) {
  19.         dataSourceKey.set(dataSource);
  20.     }
  21.     /**
  22.      * 清除dataSourceKey的值
  23.      */
  24.     public static void toDefault() {
  25.         dataSourceKey.remove();
  26.     }
  27.     /**
  28.      * 返回当前dataSourceKey的值
  29.      */
  30.     @Override
  31.     protected Object determineCurrentLookupKey() {
  32.         return dataSourceKey.get();
  33.     }
  34. }
复制代码
3.6 AOP代码
  1. package com.zhongyouex.bigdata.common.aop;
  2. import com.zhongyouex.bigdata.common.util.LoadUtil;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.aspectj.lang.JoinPoint;
  5. import org.aspectj.lang.reflect.MethodSignature;
  6. import java.lang.reflect.Method;
  7. /**
  8. * 方法拦截  粒度在mapper上(对应的sql所属xml)
  9. * @author cuizihua
  10. * @desc 切换数据源
  11. * @create 2021-09-03 16:29
  12. **/
  13. @Slf4j
  14. public class MultiDataSourceInterceptor {
  15. //动态数据源对应的key
  16.     private final String otherDataSource = "dataSourceClickhouseOther";
  17.     public void beforeOpt(JoinPoint mi) {
  18. //默认使用默认数据源
  19.         MultiDataSource.toDefault();
  20.         //获取执行该方法的信息
  21.         MethodSignature signature = (MethodSignature) mi.getSignature();
  22.         Method method = signature.getMethod();
  23.         String namespace = method.getDeclaringClass().getName();
  24. //本项目命名空间统一的规范为xxx.xxx.xxxMapper
  25.         namespace = namespace.substring(namespace.lastIndexOf(".") + 1);
  26. //这里在配置文件配置的属性为xxxMapper.ck.switch=1 or 0  1表示切换
  27.         String isOtherDataSource = LoadUtil.loadByKey(namespace, "ck.switch");
  28.         if ("1".equalsIgnoreCase(isOtherDataSource)) {
  29.             MultiDataSource.setDataSourceKey(otherDataSource);
  30.             String methodName = method.getName();
  31.         }
  32.     }
  33. }
复制代码
3.7 AOP代码逻辑说明

通过org.aspectj.lang.reflect.MethodSignature可以获取对应执行sql的xml空间名称,拿到sql对应的xml命名空间就可以获取配置文件中配置的属性决定该xml是否开启切换数据源了。
3.8 对应的aop配置
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:tx="http://www.springframework.org/schema/tx"
  4.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
  5.     <bean id="dataSourceClickhousePinpin"  destroy-method="close" lazy-init="true">
  6.         <property name="url" value="${clickhouse.jdbc.pinpin.url}" />
  7.     </bean>
  8.     <bean id="singleSessionFactoryPinpin" >
  9.         
  10. <property name="dataSource" ref="dataSourceClickhousePinpin" />
  11.     </bean>
  12. </beans>        
复制代码
以上就是整个实现过程,希望能帮上有需要的小伙伴
作者:京东物流 崔子华
来源:京东云开发者社区

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

徐锦洪

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表