我的编程空间,编程开发者的网络收藏夹
学习永远不晚

mybatisplus @DS怎么实现动态切换数据源

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

mybatisplus @DS怎么实现动态切换数据源

今天小编给大家分享一下mybatisplus @DS怎么实现动态切换数据源的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

1、mybatis-plus @DS实现动态切换数据源原理

首先mybatis-plus使用com.baomidou.dynamic.datasource.AbstractRoutingDataSource继承 AbstractDataSource接管数据源;具体实现类为com.baomidou.dynamic.datasource.DynamicRoutingDataSource。项目初始化调用public synchronized void addDataSource(String ds, DataSource dataSource)加载数据源,数据源存进dataSourceMap中。
private Map<String, DataSource> dataSourceMap = new LinkedHashMap<>();

private Map<String, DynamicGroupDataSource> groupDataSources = new ConcurrentHashMap<>();public synchronized void addDataSource(String ds, DataSource dataSource) {    if (p6spy) {      dataSource = new P6DataSource(dataSource);    }    dataSourceMap.put(ds, dataSource);    if (ds.contains(UNDERLINE)) {      String group = ds.split(UNDERLINE)[0];      if (groupDataSources.containsKey(group)) {        groupDataSources.get(group).addDatasource(dataSource);      } else {        try {          DynamicGroupDataSource groupDatasource = new DynamicGroupDataSource(group,              strategy.newInstance());          groupDatasource.addDatasource(dataSource);          groupDataSources.put(group, groupDatasource);        } catch (Exception e) {          log.error("dynamic-datasource - add the datasource named [{}] error", ds, e);          dataSourceMap.remove(ds);        }      }    }    log.info("dynamic-datasource - load a datasource named [{}] success", ds);  }

进行数据操作时,方法会被com.baomidou.dynamic.datasource.aop.DynamicDataSourceAnnotationInterceptor拦截,

public class DynamicDataSourceAnnotationInterceptor implements MethodInterceptor {    private static final String DYNAMIC_PREFIX = "#";  private static final DynamicDataSourceClassResolver RESOLVER = new DynamicDataSourceClassResolver();  @Setter  private DsProcessor dsProcessor;  @Override  public Object invoke(MethodInvocation invocation) throws Throwable {    try {      DynamicDataSourceContextHolder.push(determineDatasource(invocation));      return invocation.proceed();    } finally {      DynamicDataSourceContextHolder.poll();    }  }  private String determineDatasource(MethodInvocation invocation) throws Throwable {    Method method = invocation.getMethod();    DS ds = method.isAnnotationPresent(DS.class)        ? method.getAnnotation(DS.class)        : AnnotationUtils.findAnnotation(RESOLVER.targetClass(invocation), DS.class);    String key = ds.value();    return (!key.isEmpty() && key.startsWith(DYNAMIC_PREFIX)) ? dsProcessor        .determineDatasource(invocation, key) : key;  }}

拦截器首先从被拦截的方法或者类(一般@DS注解用于Service,也可用于Mapper和Controller)上寻找@DS注解,获取到@DS注解的值后将其存入com.baomidou.dynamic.datasource.toolkit.DynamicDataSourceContextHolder;DynamicDataSourceContextHolder使用ThreadLocal存储当前线程的数据源名。

public final class DynamicDataSourceContextHolder {    @SuppressWarnings("unchecked")  private static final ThreadLocal<Deque<String>> LOOKUP_KEY_HOLDER = new ThreadLocal() {    @Override    protected Object initialValue() {      return new ArrayDeque();    }  };  private DynamicDataSourceContextHolder() {  }    public static String peek() {    return LOOKUP_KEY_HOLDER.get().peek();  }    public static void push(String ds) {    LOOKUP_KEY_HOLDER.get().push(StringUtils.isEmpty(ds) ? "" : ds);  }    public static void poll() {    Deque<String> deque = LOOKUP_KEY_HOLDER.get();    deque.poll();    if (deque.isEmpty()) {      LOOKUP_KEY_HOLDER.remove();    }  }    public static void clear() {    LOOKUP_KEY_HOLDER.remove();  }}

进行数据操作时,会调用org.springframework.jdbc.datasource.getConnection()方法;getConnection()方法最终调用了com.baomidou.dynamic.datasource.AbstractRoutingDataSource的getConnection()方法;

  @Override  public Connection getConnection() throws SQLException {    return determineDataSource().getConnection();  }

determineDataSource()由子类com.baomidou.dynamic.datasource.DynamicRoutingDataSource实现,可以看到DynamicRoutingDataSource从DynamicDataSourceContextHolder获取数据源名称,这个在之前拦截器处理存进ThreadLocal中,如果有数据源名称则从dataSourceMap中获取,没有则获取默认的primary数据源。

public DataSource determineDataSource() {    return getDataSource(DynamicDataSourceContextHolder.peek());}public DataSource getDataSource(String ds) {    if (StringUtils.isEmpty(ds)) {        return determinePrimaryDataSource();    } else if (!groupDataSources.isEmpty() && groupDataSources.containsKey(ds)) {        log.debug("dynamic-datasource switch to the datasource named [{}]", ds);        return groupDataSources.get(ds).determineDataSource();    } else if (dataSourceMap.containsKey(ds)) {        log.debug("dynamic-datasource switch to the datasource named [{}]", ds);        return dataSourceMap.get(ds);    }    if (strict) {        throw new RuntimeException("dynamic-datasource could not find a datasource named" + ds);    }    return determinePrimaryDataSource();}private DataSource determinePrimaryDataSource() {    log.debug("dynamic-datasource switch to the primary datasource");    return groupDataSources.containsKey(primary) ? groupDataSources.get(primary)        .determineDataSource() : dataSourceMap.get(primary);}

此时的数据源已经切换成了我们需要的数据源。

数据操作完成后,方法返回第二步中的拦截器,执行DynamicDataSourceContextHolder.poll();清除掉此次的数据源,避免影响后续数据操作。

附上动态数据源相关配置

spring:  application:    name:   datasource:    dynamic:      primary: dataSource1      datasource:        dataSource1:          type: com.alibaba.druid.pool.DruidDataSource          driverClassName: com.microsoft.sqlserver.jdbc.SQLServerDriver          url: jdbc:sqlserver://localhost:1433;database=dataSource1          username:           password:         dataSource2:          type: com.alibaba.druid.pool.DruidDataSource          driverClassName: com.microsoft.sqlserver.jdbc.SQLServerDriver          url: jdbc:sqlserver://localhost:1433;instanceName=sqlserver2017;DatabaseName=dataSource2          username:           password:

pom.xml

<dependency>    <groupId>com.baomidou</groupId>    <artifactId>dynamic-datasource-spring-boot-starter</artifactId>    <version>2.5.6</version></dependency>

相应类

@Service//@DS("dataSource2") 放在类上就是类下所有方法都使用这个数据源。public class XXXServiceImpl extends BaseServiceImpl<XXXMapper, XXXBean> implements XXXService {    @DS("dataSource1")    public void selectDataFromSource1() {       // do somethinng;    }        @DS("dataSource2")    public void selectDataFromSource1() {       // do somethinng;    }}

**注意:**不可在事务中切换数据库,保证事务需要方法使用同一连接,使用@DS(dataSource1)方法调用@DS(dataSource2)无法切换连接,会导致方法报错。

以上就是“mybatisplus @DS怎么实现动态切换数据源”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

mybatisplus @DS怎么实现动态切换数据源

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

mybatisplus @DS怎么实现动态切换数据源

今天小编给大家分享一下mybatisplus @DS怎么实现动态切换数据源的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1、
2023-07-02

Springboot动态切换数据源怎么实现

这篇文章主要介绍“Springboot动态切换数据源怎么实现”,在日常操作中,相信很多人在Springboot动态切换数据源怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Springboot动态切换数
2023-06-25

java动态数据源切换怎么实现

在Java中实现动态数据源切换有多种方式,以下是其中一种常见的实现方法:1. 创建一个数据源容器类:创建一个类来管理多个数据源对象,例如一个Map,使用数据源名称作为键,数据源对象作为值。2. 创建注
2023-10-09

springboot怎么集成@DS注解实现数据源切换

这篇文章主要介绍“springboot怎么集成@DS注解实现数据源切换”,在日常操作中,相信很多人在springboot怎么集成@DS注解实现数据源切换问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”sprin
2023-06-29

Spring多数据源AOP动态切换怎么实现

这篇文章主要讲解了“Spring多数据源AOP动态切换怎么实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Spring多数据源AOP动态切换怎么实现”吧!一:新增多数据源类public c
2023-06-04

springboot+dynamicDataSource怎么实现动态添加切换数据源

今天小编给大家分享一下springboot+dynamicDataSource怎么实现动态添加切换数据源的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,
2023-06-26

SpringBoot+Mybatis如何实现动态数据源切换

这篇文章主要介绍了SpringBoot+Mybatis如何实现动态数据源切换,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。springboot是什么springboot一种全
2023-06-14

Jdbc怎么实现分布式事务数据源动态切换

本篇内容介绍了“Jdbc怎么实现分布式事务数据源动态切换”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一:依赖的jar包 Maven配置<
2023-06-04

【Java多数据源实现教程】实现动态数据源、多数据源切换方式

前言 本文为 【Java多数据源实现教程】 相关知识,由于自己最近在做导师的项目的时候需要使用这种技术,于是自学了相关技术原理与实现,并将其整理如下,具体包含:多数据源的典型使用场景(包含业务复杂场景、读写分离场景),多数据源实现原理及实
2023-08-16

MyBatisPlus怎么集成动态多数据源

本篇内容主要讲解“MyBatisPlus怎么集成动态多数据源”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“MyBatisPlus怎么集成动态多数据源”吧!这里使用的是dynamic-dataso
2023-07-02

Java注解实现动态数据源切换的实例代码

当一个项目中有多个数据源(也可以是主从库)的时候,我们可以利用注解在mapper接口上标注数据源,从而来实现多个数据源在运行时的动态切换。实现原理在Spring 2.0.1中引入了AbstractRoutingDataSource, 该类充
2023-05-31

SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换

本文小编为大家详细介绍“SpringBoot基于AbstractRoutingDataSource如何实现多数据源动态切换”,内容详细,步骤清晰,细节处理妥当,希望这篇“SpringBoot基于AbstractRoutingDataSour
2023-06-30

SpringBoot多数据源配置并通过注解实现动态切换数据源

本文主要介绍了SpringBoot多数据源配置并通过注解实现动态切换数据源,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
2022-11-13

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录