spring+mybitas 实现多数据源动态切换

简介: 1 由于项目需要,需要将不同来源的数据存入不同的数据库,所以需要根据入参的信息动态切换数据源 项目core层采用了spring+mybitas 方式进行架构 1 在spring的配置中定义两个数据源 2创建一个新的类 MultiDataSource public class Mul.

1 由于项目需要,需要将不同来源的数据存入不同的数据库,所以需要根据入参的信息动态切换数据源

项目core层采用了spring+mybitas 方式进行架构

1 在spring的配置中定义两个数据源

<bean id="dataSource2" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
 
</bean>
<bean id="dataSource1" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
 
</bean>

2创建一个新的类 MultiDataSource

public class MultiDataSource extends AbstractRoutingDataSource {
    private static final ThreadLocal<String> dataSourceKey = new InheritableThreadLocal<String>();

    public static void setDataSourceKey(String dataSource) {
        dataSourceKey.set(dataSource);
    }

    @Override
    protected Object determineCurrentLookupKey() {
        return dataSourceKey.get();
    }
}

这个类需要继承 AbstractRoutingDataSource
并且重写 determineCurrentLookupKey方法
3 在spring 的配置文件中新增配置

<bean class="com.cmos.nganocecore.util.MultiDataSource" id="dataSource">
   <property name="targetDataSources">
      <map key-type="java.lang.String">
         <entry value-ref="dataSource1" key="dataSource1"></entry>
         <entry value-ref="dataSource2" key="dataSource2"></entry>
      </map>
   </property>
   <!-- 默认使用server的数据源 -->
   <property name="defaultTargetDataSource" ref="dataSource1"></property>
</bean>

4 将配置的dataSource 添加到sessionFactory 中

<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
   <property name="configLocation" value="classpath:orm/Configuration.xml"></property>
   <property name="dataSource" ref="dataSource"></property>
</bean>

5 将sessionFactory 注入到sqlSession中

<bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">
   <constructor-arg index="0" ref="sqlSessionFactory" />
</bean>

6 动态切换数据源决定采用spring-aop 添加切点 去进行动态切换

6.1 新建数据切换类

public class DataBaseAop {
  
 

    public void before(InputObject inputObject,OutputObject outputObject) {
      if(条件){
MultiDataSource.setDataSourceKey("dataSource1");
}else{
MultiDataSource.setDataSourceKey("dataSource2");
}    

    }
}

6.2 定义数据切换类

<bean id="databaseAop" class="com.*.*.aop.DataBaseAop">

</bean>

6.3定义切点

<aop:config proxy-target-class="true">
   <aop:pointcut expression="(execution(void com.cmos.nganocecore.service.*.*.insert*(..))
     or execution(void com.cmos.nganocecore.service.*.*.query*(..))
     or execution(void com.cmos.nganocecore.service.*.*.select*(..))
     or execution(void com.cmos.nganocecore.service.*.*.find*(..))
     or execution(void com.cmos.nganocecore.service.*.*.insert*(..))
     or execution(void com.cmos.nganocecore.service.*.*.save*(..))
     or execution(void com.cmos.nganocecore.service.*.*.update*(..))
     or execution(void com.cmos.nganocecore.service.*.*.edit*(..))
     or execution(void com.cmos.nganocecore.service.*.*.modify*(..))
     or execution(void com.cmos.nganocecore.service.*.*.del*(..))
     or execution(void com.cmos.nganocecore.service.*.*.delete*(..)))

     and args(inputObject,outputObject)" id="dataBase"/>
   <aop:aspect ref="databaseAop">
      <aop:before method="before" pointcut-ref="dataBase" arg-names="inputObject,outputObject" />
   </aop:aspect>
 
</aop:config>

proxy-target-class="true" 参数是为了设置 拦截 器可以定义为class

注释
1 AbstractRoutingDataSource 类中 存在跟数据库资源建立连接的方法
public Connection getConnection()

在该方法中 调用了 protected DataSource determineTargetDataSource() 方法

在该方法中 会调用 determineCurrentLookupKey 获取dataBase 的key 并根据 key 从对应的数据源中获取对应的数据库连接资源 如数据库连接池资源
类 MultiDataSource 重写了 determineCurrentLookupKey 方法

@Override
    protected Object determineCurrentLookupKey() {
        return dataSourceKey.get();
    }

在需要切换数据源之前 调用全局方法 MultiDataSource.setDataSourceKey(String key) 去切换数据源

2

<bean class="com.*.*.util.MultiDataSource" id="dataSource">
   <property name="targetDataSources">
      <map key-type="java.lang.String">
         <entry value-ref="dataSource1" key="dataSource1"></entry>
         <entry value-ref="dataSource2" key="dataSource2"></entry>
      </map>
   </property>
   <!-- 默认使用server的数据源 -->
   <property name="defaultTargetDataSource" ref="dataSource1"></property>
</bean>

配置中的 标签中的 key 为动态切换数据源的key
defaultTargetDataSource 参数 指的是默认的数据源

相关文章
|
3月前
|
Ubuntu Java Linux
在Spring Boot中使用iTextPDF创建动态PDF文档
iTextPDF 是一个用于创建和操作 PDF(Portable Document Format)文档的流行的 Java 库。它提供了一套全面的功能,用于处理 PDF 文件,包括创建新文档、修改现有文档以及提取信息。
84 1
|
5月前
|
前端开发 Java 数据库连接
基于Spring boot轻松实现一个多数据源框架
基于Spring boot轻松实现一个多数据源框架
127 0
|
6月前
|
缓存 NoSQL Java
Spring Boot如何优雅实现动态灵活可配置的高性能数据脱敏功能
在当下互联网高速发展的时代下,涉及到用户的隐私数据安全越发重要,一旦泄露将造成不可估量的后果。所以现在的业务系统开发中都会对用户隐私数据加密之后存储落库,同时还要求后端返回数据给前台之前进行数据脱敏。所谓脱敏处理其实就是将数据进行混淆隐藏,如将用户的手机号脱敏展示为`178****5939,采用 * 进行隐藏,以免泄露个人隐私信息
173 0
|
4月前
|
前端开发 Java 数据库连接
Spring Boot 3 整合 Mybatis-Plus 动态数据源实现多数据源切换
Spring Boot 3 整合 Mybatis-Plus 动态数据源实现多数据源切换
|
4月前
|
XML Java 数据格式
spring中怎么通过静态工厂和动态工厂获取对象以及怎么通过 FactoryBean 获取对象
spring中怎么通过静态工厂和动态工厂获取对象以及怎么通过 FactoryBean 获取对象
35 0
|
11天前
|
安全 数据安全/隐私保护
Springboot+Spring security +jwt认证+动态授权
Springboot+Spring security +jwt认证+动态授权
|
30天前
|
Java 数据库连接 数据库
Spring Boot整合MyBatis Plus集成多数据源轻松实现数据读写分离
Spring Boot整合MyBatis Plus集成多数据源轻松实现数据读写分离
26 2
|
3月前
|
存储 NoSQL Java
Spring Boot动态秒杀系统接口安全性设计与实现
Spring Boot动态秒杀系统接口安全性设计与实现
55 0
|
3月前
|
Java 数据库 数据安全/隐私保护
使用Spring Boot和JPA实现多数据源的方法
使用Spring Boot和JPA实现多数据源的方法
48 0
|
3月前
|
Java 数据库连接 数据库
【Spring技术专题】「实战开发系列」保姆级教你SpringBoot整合Mybatis框架实现多数据源的静态数据源和动态数据源配置落地
Mybatis是一个基于JDBC实现的,支持普通 SQL 查询、存储过程和高级映射的优秀持久层框架,去掉了几乎所有的 JDBC 代码和参数的手工设置以及对结果集的检索封装。 Mybatis主要思想是将程序中大量的 SQL 语句剥离出来,配置在配置文件中,以实现 SQL 的灵活配置。在所有 ORM 框架中都有一个非常重要的媒介——PO(持久化对象),PO 的作用就是完成持久化操作,通过该对象对数据库执行增删改的操作,以面向对象的方式操作数据库。
44 1
【Spring技术专题】「实战开发系列」保姆级教你SpringBoot整合Mybatis框架实现多数据源的静态数据源和动态数据源配置落地