SpringBoot动态数据源配置

序:数据源动态切换流程图如下:

1:pom.xml文件依赖声明

<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
  <groupId>org.mybatis.spring.boot</groupId>
  <artifactId>mybatis-spring-boot-starter</artifactId>
  <version>2.1.4</version>
</dependency>
<dependency>
  <groupId>mysql</groupId>
  <artifactId>mysql-connector-java</artifactId>
  <version>8.0.12</version>
</dependency>
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

2:application.yml配置文件

配置多个数据源信息

server:
port: 9000

dynamic:
datasource:
  ds1: # 数据源1
    jdbc-url: jdbc:mysql://localhost:3306/test1?characterEncoding=utf8&useUnicode=true&useSSL=false&serverTimezone=GMT%2B8
    username: root
    password: 123456
    driver-class-name: com.mysql.cj.jdbc.Driver
  ds2: # 数据源2
    jdbc-url: jdbc:mysql://localhost:3306/test2?characterEncoding=utf8&useUnicode=true&useSSL=false&serverTimezone=GMT%2B8
    username: root
    password: 123456
    driver-class-name: com.mysql.cj.jdbc.Driver

mybatis:
mapper-locations: classpath:mapper/*.xml
configuration:
  map-underscore-to-camel-case: true

3:数据源的配置到Spring管理

/**
* description 数据源配置类
*
* @author liekkas 2021/08/21 12:33
*/
@Configuration
public class DynamicDataSourceConfig {

  @Bean(name = "dataSource1")
  @ConfigurationProperties(prefix = "dynamic.datasource.ds1")
  public DataSource dateSource1() {
      return DataSourceBuilder.create().build();
  }

  @Bean(name = "dataSource2")
  @ConfigurationProperties(prefix = "dynamic.datasource.ds2")
  public DataSource dateSource2() {
      return DataSourceBuilder.create().build();
  }

  @Bean
  @Primary
  public DynamicRoutingDataSource dynamicDataSource() {
      DynamicRoutingDataSource dynamicRoutingDataSource = new DynamicRoutingDataSource();

//设置默认的数据源
      dynamicRoutingDataSource.setDefaultTargetDataSource(dateSource1());
      Map<Object, Object> targetDataSources = new HashMap<>(4);

      targetDataSources.put("dataSource1", dateSource1());
      targetDataSources.put("dataSource2", dateSource2());
//设置数据源
      dynamicRoutingDataSource.setTargetDataSources(targetDataSources);
      dynamicRoutingDataSource.afterPropertiesSet();
      DynamicDataSourceContextHolder.dataSourceSet.addAll(targetDataSources.keySet());
      return dynamicRoutingDataSource;
  }
}

注意:@ConfigurationProperties注解的prefix的值和配置文件application.yml必须要保持一致

3:配置数据源上下文

我们需要新建一个数据源上下文,用户记录当前线程使用的数据源是什么,以及记录所有注册成功的数据源的集合。对于线程级别的私有变量,我们首先ThreadLocal来实现。

/**
* description 设置当前线程数据源上下文类
*
* @author liekkas 2021/08/21 12:37
*/
public class DynamicDataSourceContextHolder {
  private static final Logger logger = LoggerFactory.getLogger(DynamicDataSourceContextHolder.class);

  /**
    * 存储已经注册的数据源的key
    */
  public static Set<Object> dataSourceSet = new HashSet<>();

  /**
    * 线程级别的私有变量
    */
  private static final ThreadLocal<String> HOLDER = new ThreadLocal<>();

  public static String getDataSourceRouterKey() {
      return HOLDER.get();
  }

  public static void setDataSourceRouterKey(String dataSourceRouterKey) {
      logger.info("当前数据源正在切换中......");
      HOLDER.set(dataSourceRouterKey);
  }

  /**
    * 设置数据源之前一定要先移除
    */
  public static void removeDataSourceRouterKey() {
      HOLDER.remove();
  }

  /**
    * 判断指定dataSource当前是否存在
    *
    * @param dataSource dataSource
    * @return boolean
    */
  public static boolean isExistDataSource(Object dataSource) {
      return dataSourceSet.contains(dataSource);
  }

}

4:动态数据源路由

第三步我们以及新建了数据源上下文,用于存储我们当前线程的数据源,那么怎么通知spring用当前的数据源,spring提供一个接口,名为AbstractRoutingDataSource的抽象类,我们只需要重写determineCurrentLookupKey方法就可以,这个方法看名字就知道,就是返回当前线程的数据源,那我们只需要从我们刚刚的数据源上下文中取出当前数据源即可

/**
* description 数据源动态路由
*
* @author liekkas 2021/08/21 12:39
*/
public class DynamicRoutingDataSource extends AbstractRoutingDataSource {

  private static final Logger logger = LoggerFactory.getLogger(DynamicRoutingDataSource.class);

  @Override
  protected Object determineCurrentLookupKey() {
      String dataSourceName = DynamicDataSourceContextHolder.getDataSourceRouterKey();
      logger.info("当前数据源是:{}", dataSourceName);
      return DynamicDataSourceContextHolder.getDataSourceRouterKey();
  }
}:

5:数据源动态切换注解

/**
* description 动态切换数据源注解
* 该注解可以同时作用于类和方法上,如果类和方法注解同时存在,则方法的注解优先级高于类上的注解优先级
* 也就是说类和方法同时存在该注解的话,则会使用方法的数据源
*
* @author liekkas 2021/08/21 12:40
*/
@Target({ElementType.METHOD, ElementType.TYPE, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface DataSource {
  String value() default "dataSource1";
}:

6:利用aop实现数据源动态切换

/**
* description 动态切换数据源切面
*
* @author liekkas 2021/08/21 12:42
*/
@Aspect
@Component
@Order(-1)
public class DynamicDataSourceAspect {
  private static final Logger logger = LoggerFactory.getLogger(DynamicDataSourceAspect.class);

  /**
    * 同时拦截标注DataSource的注解类上或者方法
    */
  @Pointcut(value = "@within(com.liekkas.config.datasource.DataSource) || @annotation(com.liekkas.config.datasource.DataSource)")
  public void datasource() {
  }


  @Before("datasource()")
  public void changeDataSource(JoinPoint joinPoint) {

      Class<?> klass = joinPoint.getTarget().getClass();

      MethodSignature signature = (MethodSignature) joinPoint.getSignature();
      Method method = signature.getMethod();

      DataSource dataSource;
      if (method.isAnnotationPresent(DataSource.class)) {
          dataSource = method.getAnnotation(DataSource.class);
      } else {
          dataSource = klass.getAnnotation(DataSource.class);
      }

      //判断注解中的数据源是否在已注册的数据源中
      if (DynamicDataSourceContextHolder.isExistDataSource(dataSource.value())) {
          DynamicDataSourceContextHolder.setDataSourceRouterKey(dataSource.value());
          logger.info("数据源设置为 > {}", dataSource.value());
      } else {
          throw new RuntimeException("数据源[" + dataSource.value() + "]不存在");
      }
  }

  @After("datasource()")
  public void restoreDataSource() {
      DynamicDataSourceContextHolder.removeDataSourceRouterKey();
  }
}

7:测试

7.1:controller

/**
* description
*
* @author liekkas 2021/01/09 15:47
*/
@RestController
public class PersonController {

  @Resource
  private PersonService personService;

  @GetMapping("person")
  public List<Person> findPerson() {
      return personService.findPerson();
  }

  @GetMapping("student")
  public List<Student> findStudent() {
      return personService.findStudent();
  }
}

7.2: service

public interface PersonService {
  /**
    * findPerson
    *
    * @return list
    */
  List<Person> findPerson();
  /**
    * findStudent
    *
    * @return list
    */
  List<Student> findStudent();
}

7.3: serviceImpl

@Service
@DataSource("dataSource1")
public class PersonServiceImpl implements PersonService {

  @Resource
  private PersonMapper personMapper;

  @Override
  public List<Person> findPerson() {
      return personMapper.findPerson();
  }

  @Override
  @DataSource("dataSource2")
  public List<Student> findStudent() {
      return personMapper.findStudent();
  }
}

由于篇幅关系,mapper以及实体类不再描述

测试接口调用

http://localhost:9000/person

http://localhost:9000/student

至此数据源动态切换已经完成。

小结:数据源动态切换主要原理就是实现了Spring提供的AbstractRoutingDataSource类,然后我们就可以根据自己的需要利用aop和自定义注解,ThreadLocal来实现动态切换使用哪一个数据源了。

SpringBoot动态数据源配置的更多相关文章

  1. Springboot+Druid 动态数据源配置监控

    一.引入maven依赖,使用 starter 与原生 druid 依赖配置有所不同 <dependency> <groupId>com.alibaba</groupId& ...

  2. SpringBoot动态数据源

    1.原理图 2.创建枚举类 /** * 存数据源key值 */ public enum DataSourceKey { master,salve,migration } 3.创建自定义注解类 /** ...

  3. Druid动态数据源配置

    上文已经讲了单个数据源的Druid的配置(http://www.cnblogs.com/nbfujx/p/7686634.html) Druid动态数据源配置 主要是继承AbstractRouting ...

  4. Spring Boot2.x 动态数据源配置

    原文链接: Spring Boot2.x 动态数据源配置 基于 Spring Boot 2.x.Spring Data JPA.druid.mysql 的动态数据源配置Demo,适合用于数据库的读写分 ...

  5. Springboot 多数据源配置,结合tk-mybatis

    一.前言 作为一个资深的CRUD工程师,我们在实际使用springboot开发项目的时候,难免会遇到同时使用多个数据库的情况,比如前脚刚查询mysql,后脚就要查询sqlserver. 这时,我们很直 ...

  6. Spring-Boot 多数据源配置+动态数据源切换+多数据源事物配置实现主从数据库存储分离

    一.基础介绍 多数据源字面意思,比如说二个数据库,甚至不同类型的数据库.在用SpringBoot开发项目时,随着业务量的扩大,我们通常会进行数据库拆分或是引入其他数据库,从而我们需要配置多个数据源. ...

  7. Springboot多数据源配置--数据源动态切换

    在上一篇我们介绍了多数据源,但是我们会发现在实际中我们很少直接获取数据源对象进行操作,我们常用的是jdbcTemplate或者是jpa进行操作数据库.那么这一节我们将要介绍怎么进行多数据源动态切换.添 ...

  8. springboot+多数据源配置

    作者:纯洁的微笑 出处:http://www.ityouknow.com/ 起多数据源,一般都来解决那些问题呢,主从模式或者业务比较复杂需要连接不同的分库来支持业务.我们项目是后者的模式,网上找了很多 ...

  9. Spring Boot + Mybatis多数据源和动态数据源配置

    文章转自 https://blog.csdn.net/neosmith/article/details/61202084 网上的文章基本上都是只有多数据源或只有动态数据源,而最近的项目需要同时使用两种 ...

  10. 基于注解实现SpringBoot多数据源配置

    1.功能介绍 在实际的开发中,同一个项目中使用多个数据源是很常见的场景.最近在学习的过程中使用注解的方式实现了一个Springboot项目多数据源的功能.具体实现方式如下. 2.在applicatio ...

随机推荐

  1. Win10下小米路由器4A百兆版刷Openwrt固件【图片详细版】

    将原来的小米路由器换成了华为,早就听闻刷软路由可以实现去广告,解锁灰色歌单等诸多骚操作.就来榨取这个小米4A的剩余价值来着的. 注意 1. 必须使用路由模式,中继模式是打不开telnet的 更新固件 ...

  2. FPGA最大工作频率教程

    FPGA最大工作频率教程 1.  Quartus的时序分析 作为编译过程的一部分,Quartus对布局布线的电路做时序分析.在编译报告里,展开"Timing Analyzer",这 ...

  3. 3种方法实现图片瀑布流的效果(纯JS,Jquery,CSS)

    最近在慕课网上听如何实现瀑布流的效果:介绍了3种方法. 1.纯JS代码实现: HTML代码部分: <!DOCTYPE html> <html> <head> < ...

  4. 我的书《Unity3D动作游戏开发实战》出版了

    首先感谢帮助和参与前期检阅的朋友们.本书是我经验积累的提炼,书中既有干货分享也有对基础内容的详解补充. 同时由于是第一次撰写书籍,许多地方仍有不足还请读者朋友们见谅. 在京东或当当等都可以购买到本书: ...

  5. 提取关键词作为标题---Java调用Python实现

    目录 前景提示 思考 企业级处理方案流程图 Python 实现的方式 Java实现 导入Maven包 书写代码 最终版本 遗留问题: 遇到问题 1.无法解析. 2.模块找不到. 3.乱码处理 版本 继 ...

  6. 用 C 语言开发一门编程语言 — 抽象语法树

    目录 文章目录 目录 前文列表 抽象语法树的结构 使用递归来遍历树结构 实现求值计算 抽象语法树与行为树 前文列表 <用 C 语言开发一门编程语言 - 交互式解析器l> <用 C 语 ...

  7. 智能便捷_AIRIOT智慧充电桩管理解决方案

    现如今随着对可持续交通的需求不断增加,电动车市场正在迅速扩大,建设更多更智能的充电桩,并通过管理平台提高充电设施的可用性和效率成为一项重要任务.传统的充电桩管理平台在对充电设施进行管理过程中,存在如下 ...

  8. python ddddocr图片验证码详解

    安装 下载地址:https://pypi.tuna.tsinghua.edu.cn/simple/ddddocr/ 安装命令: pip install D:\ChromeCoreDownloads\d ...

  9. 在WPF中判断是是否为设计时模式

    方式一:   using System.ComponentModel;       private bool IsInDesignMode   {   get { return DesignerPro ...

  10. 莫烦pytorch学习记录

    感谢莫烦大神Pytorch B站视频:https://www.bilibili.com/video/av15997678?p=11 一个博主的笔记:https://blog.csdn.net/Will ...