mybatis-plus动态数据源切换不生效的问题解决

2023-05-16

一、问题描述
在我们项目中,既要连接mysql,又要连接TDEngine(taos),正确配置后也无法动态切换数据源执行sql
二、环境
1.依赖

<!--连接另外一种数据库的驱动-->
        <dependency>
            <groupId>com.taosdata.jdbc</groupId>
            <artifactId>taos-jdbcdriver</artifactId>
<!--            <version>2.0.32</version>-->
            <version>3.0.0</version>
        </dependency>
        <!--mybatis plus 动态切换数据源的依赖-->
   		<dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
            <version>3.3.2</version>
        </dependency>

2.配置

spring:
  datasource:
    dynamic:
      strict: true
      primary: mysql
      datasource:
        mysql:
          driver-class-name: com.mysql.cj.jdbc.Driver
          url: jdbc:mysql://192.xxx.xxx.xxx:3306/db_iot?useUnicode=true&characterEncoding=UTF-8&useSSL=false&serverTimezone=Asia/Shanghai
          username: root
          password: 123456
        tdengine:
          driver-class-name: com.taosdata.jdbc.TSDBDriver
          url: jdbc:TAOS://192.xxx.xxx.xxx:6030/iot_data?timezone=UTC-8&charset=UTF-8&locale=en_US.UTF-8
          username: root
          password: taosdata
        mysql1:
          driver-class-name: com.mysql.cj.jdbc.Driver
          url: jdbc:mysql://192.xxx.xxx.xxx:3306/db_portal?useUnicode=true&characterEncoding=UTF-8&useSSL=false&serverTimezone=Asia/Shanghai
          username: root
          password: 123456



三、解决方法
直接上代码,最后会有问题分析,因为涉及到源码,这里暂时不讲
这里的dynamicRoutingDataSource()不能有DynamicDataSourceAutoConfiguration里的那样命名和返回值,那样在注入时就会因为其他数据源的注入导致无法注入动态数据源,需要细化到具体的类型

package com.xxx.project.iotconf.configs;

import com.baomidou.dynamic.datasource.DynamicRoutingDataSource;
import com.baomidou.dynamic.datasource.provider.DynamicDataSourceProvider;
import com.baomidou.dynamic.datasource.provider.YmlDynamicDataSourceProvider;
import com.baomidou.dynamic.datasource.spring.boot.autoconfigure.DataSourceProperty;
import com.baomidou.dynamic.datasource.spring.boot.autoconfigure.DynamicDataSourceProperties;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;

import javax.sql.DataSource;
import java.util.Map;


@Configuration
public class TDEngineConfig {

    @Autowired
    private DynamicDataSourceProperties properties;

    @Bean
    @ConditionalOnMissingBean
    @Primary
    public DynamicDataSourceProvider dynamicDataSourceProvider() {
        Map<String, DataSourceProperty> datasourceMap = this.properties.getDatasource();
        return new YmlDynamicDataSourceProvider(datasourceMap);
    }

    @Bean
    @ConditionalOnMissingBean
    public DynamicRoutingDataSource dynamicRoutingDataSource(DynamicDataSourceProvider dynamicDataSourceProvider) {
        DynamicRoutingDataSource dynamicRoutingDataSource = new DynamicRoutingDataSource();
        dynamicRoutingDataSource.setPrimary(this.properties.getPrimary());
        dynamicRoutingDataSource.setStrict(this.properties.getStrict());
        dynamicRoutingDataSource.setStrategy(this.properties.getStrategy());
        dynamicRoutingDataSource.setProvider(dynamicDataSourceProvider);
        dynamicRoutingDataSource.setP6spy(this.properties.getP6spy());
        dynamicRoutingDataSource.setSeata(this.properties.getSeata());
        Map<String, DataSource> dataSourceMap = dynamicDataSourceProvider.loadDataSources();
        for (String key : dataSourceMap.keySet()) {
            dynamicRoutingDataSource.addDataSource(key, dataSourceMap.get(key));
        }
        return dynamicRoutingDataSource;
    }

}

四、测试
放入TAOS创建超表的SQL,由JdbcTemplate去执行,执行成功
在这里插入图片描述

package com.xxx.project.iot.pulsar.handler;

import com.baomidou.dynamic.datasource.annotation.DS;
import com.shandy.project.api.basic.dto.SDevice;
import com.shandy.project.api.basic.dto.Tuple2;
import com.shandy.project.iot.pulsar.utils.ReflectUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Service;

import javax.annotation.PostConstruct;
import java.util.List;
import java.util.Map;

@Service
@Slf4j
public class DataHandler implements DsHandler{

    @Autowired
    private JdbcTemplate jdbcTemplate;

    @PostConstruct
    public void init() {

    }

    /**
     * 执行单条DDL sql
     */
    @DS(value = "tdengine")
    public void execute(String sql) {
        jdbcTemplate.execute(sql);
        log.info(sql);
    }

    /**
     * 执行单条DML sql
     */
    @DS(value = "tdengine")
    public List<Map<String, Object>> query(String sql, Object[] args) {
        log.info(String.format("%s,params is {%s}", sql, args));
        return jdbcTemplate.queryForList(sql, args);
    }

    /**
     * 保存设备数据(批量,多表多条)
     */
    @DS(value = "tdengine")
    public void batchInsertDevice(List<SDevice> devices, String table) {
        String[] sqls = null;
        jdbcTemplate.batchUpdate(sqls);
        log.info(sqls.toString());
    }

    /**
     * 单条插入
     *
     * @param device  消息结构体
     * @param mqttObj 模型对象,IMqttR或IMqttS对象
     */
    @DS(value = "tdengine")
    public void insertDevice(SDevice device, Object mqttObj) {
        Tuple2<String, List<Object>> t2 = ReflectUtils.getInsSql(mqttObj.getClass(), device);
        List<Object> list = t2.getField(1);
        String sql = t2.getField(0);
        jdbcTemplate.update(sql, list.toArray());
        log.info(sql);
    }
}

五、问题分析
1.一开始执行时,总是报错误的SQL语句,要我检查Mysql的版本,所以从这个提示来看,是没有动态切换到我们的taos数据库的。
什么原因呢?我们看com.baomidou.dynamic.datasource.spring.boot.autoconfigure.DynamicDataSourceAutoConfiguration类的方法

   @Bean
    @ConditionalOnMissingBean
    public DataSource dataSource(DynamicDataSourceProvider dynamicDataSourceProvider) {
        DynamicRoutingDataSource dataSource = new DynamicRoutingDataSource();
        dataSource.setPrimary(this.properties.getPrimary());
        dataSource.setStrict(this.properties.getStrict());
        dataSource.setStrategy(this.properties.getStrategy());
        dataSource.setProvider(dynamicDataSourceProvider);
        dataSource.setP6spy(this.properties.getP6spy());
        dataSource.setSeata(this.properties.getSeata());
        return dataSource;
    }

这里有个@ConditionalOnMissingBean注解,意思是当dataSource对象不存在时才会进行注入。
我发现我除了配置了动态数据源,也配置了druid数据源,在项目启动是肯定是会注入druid的DataSource对象的,那就导致我们的动态数据源的DataSource对象无法注入,可能这就是切换不了的原因。这里或许可以尝试把druid数据源去掉,但我没有往这个方向去深究。

spring.datasource.druid.db-type=mysql
spring.datasource.druid.driver-class-name=com.mysql.cj.jdbc.Driver
spring.datasource.druid.url=jdbc:mysql://192.xxx.xxx.xxx:3306/db_iot?useUnicode=true&characterEncoding=UTF-8&useSSL=false&serverTimezone=Asia/Shanghai
spring.datasource.druid.username=root
spring.datasource.druid.password=123456

既然没有注入DynamicRoutingDataSource对象,我自己就写了上文中的TDEngineConfig去注入,其中方法和DynamicDataSourceAutoConfiguration类似,只是稍微做改造。

在最开始我原封不动的把DynamicDataSourceAutoConfiguration方法抄下来,在执行SQL时报:dynamic-datasource could not find a datasource named tdengine
异常来自于在DynamicRoutingDataSource的getDataSource()方法

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

从这里发现,DynamicRoutingDataSource的dataSourceMap是空的,那自然报错,但是DynamicDataSourceProperties的dataSourceMap并不是空的,所以配置并没有出错,只是项目在启动时没有填充DynamicRoutingDataSource的dataSourceMap,那不妨我们自己来完成这件事情。
在TDEngineConfig的dynamicDataSourceProvider()方法中将properties的datasourceMap封装到YmlDynamicDataSourceProvider中,再看它的loadDataSources(),不就可以获取到Map<String, DataSource>类型的一个对象吗,我们把这个想办法赋值给DynamicRoutingDataSource的dataSourceMap

   public Map<String, DataSource> loadDataSources() {
        return this.createDataSourceMap(this.dataSourcePropertiesMap);
    }

所以在TDEngineConfig的dynamicRoutingDataSource方法中通过如下代码获取到dataSourceMap

 Map<String, DataSource> dataSourceMap = dynamicDataSourceProvider.loadDataSources();

再通过下面代码就将DynamicRoutingDataSource的dataSourceMap填充好了

 for (String key : dataSourceMap.keySet()) {
            dynamicRoutingDataSource.addDataSource(key, dataSourceMap.get(key));
        }

此时完成DynamicRoutingDataSource的注入。
这样再调用SQL时,就可以获取@DS注解的value值,充当key去dataSourceMap里找到对应的数据源进行切换

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

mybatis-plus动态数据源切换不生效的问题解决 的相关文章

  • 技术面面试高频考点总结-Redis篇

    文章目录 技术面面试高频考点总结 Redis篇一 Redis常考考点枚举二 Redis推荐学习资料三 小结 技术面面试高频考点总结 Redis篇 大家好呀 xff0c 这里是小黛 xff01 Redis是目前非常火的 xff0c 完全开源的
  • HR面与反问环节考点总结

    文章目录 HR面与反问环节考点总结1 HR面1 1 HR面的重要性与注意事项1 2 HR面常考问题枚举 2 反问2 1 反问是什么2 2 反问都问些什么 小结题外话 HR面与反问环节考点总结 大家好呀 xff0c 这里是小黛 xff01 本
  • 技术面面试高频考点总结-Linux篇

    技术面面试高频考点总结 Linux篇 大家好呀 xff0c 这里是小黛 xff01 Linux是一种类Unix操作系统 xff0c 是由芬兰的林纳斯 托瓦兹 xff08 Linus Torvalds xff09 在1991年首次发布 Lin
  • 技术面面试高频考点总结-操作系统篇

    技术面面试高频考点总结 操作系统篇 文章目录 技术面面试高频考点总结 操作系统篇一 操作系统必考考点列举二 操作系统推荐学习资料三 小结题外话 大家好呀 xff0c 这里是小黛 xff01 操作系统也是在面试中必考的内容 xff0c 那今天
  • 面试高频考点总结-云原生、微服务、K8s、分布式

    文章目录 技术面面试高频考点总结 云原生篇一 云原生简介二 云原生面试考题列举三 小结题外话 技术面面试高频考点总结 云原生篇 大家好呀 xff0c 这里是小黛 xff01 之前的文章呀 xff0c 也介绍了大部分计算机基础相关的八股文 x
  • 技术面面试高频考点总结-JAVA篇

    文章目录 技术面面试高频考点总结 JAVA篇一 Java基础二 JVM三 并发四 框架五 小结题外话 技术面面试高频考点总结 JAVA篇 大家好呀 xff0c 这里是小黛 xff01 今天呀 xff0c 来介绍大家最喜欢的Java篇 xff
  • 视图绑定【android jetpack】

    视图绑定 视图绑定 是通过对xml属性转化为对应的绑定类 会对xml布局中所有的具有id属性的组件创建直接的引用 视图绑定 与 传统的findviewbyid 视图绑定 xff1a 是对id属性组件的直接引用 相当于只要组件只要有了id属性
  • NLTK语料库下载

    仓库地址 https github com Trkly NLTK DATA 希望能够帮到大家并希望大家给个star 语料库的存放位置参考如下图
  • Manjaro入坑

    Manjaro入坑 没有技术还偏偏就是喜欢瞎折腾 xff0c 没错 xff0c 说的就是我 xff01 大一的时候装过 Ubuntu 43 Win10 以及 CentOS7 8 43 Win10 的双操作系统 xff0c 后来因为一些日常软
  • android 监听手机电量变化

    今天 xff0c 简单讲讲如何监听手机电量的变化 监听电量是不能静态注册的 后来上网搜索 xff0c 发现有五个不能静态注册的广播 xff0c 这里记录一下 xff0c 免得下次再后知后觉的发现并惊讶于自己的笨拙 不能静态注册的广播 and
  • 【笔记】libpcap安装与使用

    Ubuntu 20 04 文章目录 起步一 下载libpcap 库二 安装三 使用 完整程序 们 1 头文件集锦myheader h 2 打印报文内容sniff c 3 嗅探与伪造icmpspoof c 附加内容unmask 0 共享内存报
  • 2022基于Anaconda环境下Jupyter最全安装插件及配置相关知识及踩坑指南

    2022Jupyter 43 Anaconda最全安装相关知识及踩坑指南 一 简介1 概述2 下载网址以及参考文档3 优点 二 安装使用过程1 起始2 更改存储路径 xff08 踩坑 xff08 1 xff09 xff09 2 1 使用管理
  • 微信小程序整合WebScoket部署上线

    微信小程序整合WebScoket部署上线 在部署的服务器 中安装nginx xff0c 并作出如下配置 xff1a 思路 xff1a 此处监听的是外部的443端口和xcxapi wfeifei com域名 xff0c 当通过该域名和端口号进
  • Vue(v-show简介)

    v show简介 1 v show指令的作用是 xff1a 根据切换元素的显示状态 2 原理是修改元素 的display xff0c 实现显示隐藏 3 指令后面的内容 xff0c 最终都会解析为布尔值 4 值为true元素显示 xff0c
  • 使用python编写adb脚本实现自动化测试

    使用python编写adb脚本实现自动化测试 前言一 python与adb二 使用步骤1 基本原理2 方法1 xff1a 使用os system 函数实现2 方法2 xff1a 使用os popen 函数实现2 方法3 xff1a 使用su
  • 怎么在jupyter中安装tensorflow,并切换环境为tensorflow

    首先说一下遇到的问题是如下 xff08 前提是已经安装anaconda xff09 xff1a 一 进入jupyter打开终端 在cmd中输入jupyter notebook进入jupyter xff1a 然后进入终端 xff1a 进入之后
  • selenium 自动化测试

    selenium 是最广泛使用的开源Web UI xff08 用户界面 xff09 自动化测试套件之一 selenium测试脚本可以使用任何支持的编程语言进行编码 xff0c 并且可以直接在大多数现代Web浏览器中运行 在爬虫领域selen
  • 1.spring框架-----spring framework bean基础配置 bean生命周期

    目录 一 spring framework 1 IoC控制反转 Inversion of Control 2 DI依赖注入 Dependency Injection 3 bean基础配置 4 bean实例化 4 1构造方法 4 2静态工厂
  • 华为FusionCompute资源配置

    文章目录 前言主机配置添加主机网口绑定 网络资源配置创建DVS添加上行链路添加VLAN池创建端口组 存储资源配置添加存储接口添加存储资源 前言 在上一篇文章 xff1a 在VMware Workstation和云服务器上安装华为Fusion
  • 【无标题】javaweb文件上传与下载

    构建数据库filetest xff1a 包含文件名称及路径 SET FOREIGN KEY CHECKS 61 0 Table structure for file DROP TABLE IF EXISTS 96 file 96 CREAT

随机推荐