Mybatis-Plus&&Druid多数据源配置

2023-11-18

多数据源配置思路

  • yml中配置多个数据源;
  • 通过AOP自动切换不同的数据源;
  • 配合Mybatis-plus使用

yml配置

 

spring:
  datasource:
    druid:
      db1:
        url: jdbc:mysql://10.168.1.118:3306/rd?useUnicode=true&characterEncoding=utf-8&useSSL=false
        username: root
        password: 123456
        driver-class-name: com.mysql.cj.jdbc.Driver
        #初始连接数 默认0
        initial-size: 10
        #最大连接数,默认8
        max-active: 30
        #最小闲置数
        min-idle: 10
        #获取连接的最大等待时间,单位毫秒
        max-wait: 2000
      db2:
        url: jdbc:postgresql://127.0.0.1:5432/postgres
        username: postgres
        password: 123456
        driver-class-name: org.postgresql.Driver
        #初始连接数 默认0
        initial-size: 10
        #最大连接数,默认8
        max-active: 30
        #最小闲置数
        min-idle: 10
        #获取连接的最大等待时间,单位毫秒
        max-wait: 2000

mybatis-plus:
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl # 打印SQL
  mapper-locations: classpath:mappers/**/*.xml

启动多个数据源  

@EnableTransactionManagement
@Configuration
@MapperScan("com.kuiper.mapper")
public class MybatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor paginationInterceptor() {
        MybatisPlusInterceptor interceptor  = new MybatisPlusInterceptor();
        interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
        return interceptor;
    }

    /**
     * mysql数据源
     *
     * @return
     */
    @Bean("db1DataSource")
    @ConfigurationProperties(prefix = "spring.datasource.druid.db1")
    public DataSource db1DataSource() {
        return DruidDataSourceBuilder.create().build();
    }

    /**
     * postgresql数据源
     *
     * @return
     */
    @Bean("db2DataSource")
    @ConfigurationProperties(prefix = "spring.datasource.druid.db2")
    public DataSource db2DataSource() {
        return DruidDataSourceBuilder.create().build();
    }

    /**
     * @description: 动态数据源配置
     * @param: db1DataSource mysql数据源,db2DataSource postgresql数据源
     * @return:
     * @author czl
     * @date: 2021/6/16 11:02
     */
    @Bean
    @Primary
    public DataSource multipleDataSource(@Qualifier("db1DataSource") DataSource db1DataSource, @Qualifier("db2DataSource") DataSource db2DataSource) {
        /** 创建动态数据源决策者 */
        DynamicDataSource dynamicDataSource = new DynamicDataSource();
        /** 存放多个数据源 */
        Map<Object, Object> targerDataSource = new HashMap<>(16);
        targerDataSource.put(DataSourceTypeEnum.db1DataSource.getValue(), db1DataSource);
        targerDataSource.put(DataSourceTypeEnum.db2DataSource.getValue(), db2DataSource);
        /** 讲多个数据源注入targetDataSources */
        dynamicDataSource.setTargetDataSources(targerDataSource);
        /** 默认数据源 */
        dynamicDataSource.setDefaultTargetDataSource(db1DataSource);
        return dynamicDataSource;
    }

    @Bean("sqlSessionFactory")
    public SqlSessionFactory sqlSessionFactory() throws Exception {
        /** 创建mybatis中的sqlSessionFactoryBean工厂 */
        MybatisSqlSessionFactoryBean sqlSessionFactory = new MybatisSqlSessionFactoryBean();
        /** 向sqlSessionFactoryBean工厂中注入数据源 */
        sqlSessionFactory.setDataSource(this.multipleDataSource(db1DataSource(), db2DataSource()));
        /** 创建MybatisConfiguration */
        MybatisConfiguration mybatisConfiguration = new MybatisConfiguration();
        mybatisConfiguration.setJdbcTypeForNull(JdbcType.NULL);
        /** 开启驼峰命名规则 */
        mybatisConfiguration.setMapUnderscoreToCamelCase(true);
        /** 是否开启缓存 */
        mybatisConfiguration.setCacheEnabled(false);
        sqlSessionFactory.setConfiguration(mybatisConfiguration);
        return sqlSessionFactory.getObject();
    }
}

枚举DataSourceTypeEnum类 

public enum DataSourceTypeEnum {

    /**
     * mysql数据源源
     */
    db1DataSource("db1DataSource"),
    /**
     * postgresql数据源
     */
    db2DataSource("db2DataSource");

    private String value;

    DataSourceTypeEnum(String value) {
        this.value = value;
    }

    public String getValue() {
        return value;
    }
}

动态决策数据源

@Slf4j
public class DynamicDataSource extends AbstractRoutingDataSource {

    /**
     * @description:  动态数据源决策
     * @param: null
     * @return:
     * @author
     * @date: 2021/6/16 11:09
     */
    @Override
    protected Object determineCurrentLookupKey() {
        String dataSourceType = DataSourceContextHolder.getDataSourceType();
        log.info("当前数据源为:{}", dataSourceType);
        return dataSourceType;
    }
}

 设置、添加、获取数据源

public class DataSourceContextHolder {


    private static final ThreadLocal CONTEXT_HOLDER = new ThreadLocal<>();

    /**
     * 设置数据源
     *
     * @param dataSourceTypeEnum
     */
    public static void setDataSourceType(DataSourceTypeEnum dataSourceTypeEnum) {
        CONTEXT_HOLDER.set(dataSourceTypeEnum.getValue());
    }

    /**
     * 取得当前数据源
     *
     * @return
     */
    public static String getDataSourceType() {
        return (String) CONTEXT_HOLDER.get();
    }

    /**
     * 清除上下文数据
     *
     * @return
     */
    public static void clearDataSourceType() {
        CONTEXT_HOLDER.remove();
    }
}

 AOP实现动态数据源的切换

@Component
@Order(-100)
@Slf4j
@Aspect
public class DataSourceAspect {

    @Pointcut("execution(* com.kuiper.mapper.db1..*.*(..))")
    private void db1Aspect() {
    }

    @Pointcut("execution(* com.kuiper.mapper.db2..*.*(..))")
    private void db2Aspect() {
    }

    @Before("db1Aspect()")
    public void db1DataSource() {
        log.info("切换数据源为:{}", DataSourceTypeEnum.db1DataSource.getValue());
        DataSourceContextHolder.setDataSourceType(DataSourceTypeEnum.db1DataSource);
    }

    @Before("db2Aspect()")
    public void db2DataSource() {
        log.info("切换数据源为:{}", DataSourceTypeEnum.db2DataSource.getValue());
        DataSourceContextHolder.setDataSourceType(DataSourceTypeEnum.db2DataSource);
    }

}

 项目工程结构

跨域请求配置

@Configuration
public class WebMvcConfig implements WebMvcConfigurer {

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        /** 设置允许跨域路径 */
        registry.addMapping("/**")
                /** 设置允许跨域请求的域名 */
                .allowedOriginPatterns("*")
                /** 是否允许证书 不在默认开启 */
                .allowCredentials(true)
                /** 设置允许的方法 */
                .allowedMethods("*")
                /** 跨域允许时间 */
                .maxAge(3600);
    }
}

 

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

Mybatis-Plus&&Druid多数据源配置 的相关文章

随机推荐

  • 使用 PullToRefresh 的总结

    前言 关于下拉刷新 上拉加载的框架现在有很多 这里奉上别人收集的一些框架 下拉刷新框架收集 但是笔者一直还在使用 PullToRefresh 个人觉得 PullToRefresh 使用起来还是比较简洁方便的 关于 PullToRefresh
  • unity-ugui-eventsystem

    EventSystem对象的说明 当我们在场景中创建任一UI对象后 Hierarchy面板中都可以看到系统自动创建了对象EventSystem 可以看到该对象下有三个组件 EventSystem StandaloneInputModule
  • 七天引爆社交新零售(助你提高十倍业绩)——前言

    2019年对于中小企业主 创业者 实体店主最大的机遇就是社交新零售 为什么这么说呢 随着日益上涨房租成本 人工成本的上升 实体生意利润空间越来越小了 而传统线上电商企业流量广告费也越来越贵了 大家一直在探索有没有一种低成本高收益的销售方式出
  • stream()转map转list、distinct()去重、判断空值、sorted排序正序多字段排序

    package demo io import demo api JavaBean Student import org junit platform commons util StringUtils import java util imp
  • 镜头快速精准反馈位置硬件环境搭建

    目录 概述 一 检测部分 1 原理图 2 PCB板 二 驱动部分 1 原理图 2 PCB板 概述 本篇只要介绍 硬件电路搭建 这次是项目的需要 重新捡起好多年没使用 Altium Designer 软件了 熟悉又陌生 经过2 3天时间 终于
  • 想要精通算法和SQL的成长之路 - 最长回文子序列

    想要精通算法和SQL的成长之路 最长回文子序列 前言 一 最长回文子序列 前言 想要精通算法和SQL的成长之路 系列导航 一 最长回文子序列 原题链接 首先 我们看下动态规划方程的定义 我们用dp i j 来代表 字符串s在下标区间为 i
  • Django连接MySQL数据库时出错:django.core.exceptions.ImproperlyConfigured: Error loading MySQLdb module: No mo

    基于python3解释器的虚拟环境中创建的Django项目 Django中默认连接的是SQLite3数据库 现更改为MySQL数据库 执行迁移文件时报错 django core exceptions ImproperlyConfigured
  • html5 调用本地街景,H5案例分享:在移动端调用腾讯街景

    在移动端调用腾讯街景 腾讯地图街景组件可以通过多种方式调起 来展示3D街景信息 腾讯街景API 是构建在v2版本上的全新应用接口 对于目的地 可以让用户足不出户 得到更直观 更真切 的身临其境的体验 比如 您可以就用在 房产 酒店 餐饮 娱
  • 使用Java 8函数式编程生成字母序列

    在 Java 8 中使用函数式编程生成字母序列是一个很大的挑战 Lukas Eder 愉快地接受了这个挑战 他将告诉我们如何使用 Java 8 来生成ABC的序列 当然 肯定不是一种蹩脚的方式 我被 Stack Overflow 上网友 m
  • C++ xml库的选择

    自从触及xml文件的读写 一直以来都是用的tinyxml2 接口简单 然而近期项目频繁出错 跟踪调试发现 问题出在了xml文件的读写上 当节点数超过百万级别的时候 内存暴增到G的当量 很显然程序会由于内存申请不足崩掉了 果断寻找替代品 百度
  • Android开源框架之Picasso(图片加载框架)

    简介 Picasso是Square公司出品的一个强大的图片下载和缓存图片库 在adapter中需要取消已经不在视野范围的ImageView图片资源的加载 否则会导致图片错位 Picasso已经解决了这个问题 使用复杂的图片压缩转换来减少内存
  • ue4 蓝图通信的几种方式

    一 设置公有变量 完成通信 1 蓝图类Door bp中声明变量NewVar 1 为公有 确定好变量类型 编译 2 关卡视口中选中这个蓝图类Door bp的实例 世界大纲视图下的细节面板中 默认下出现公有变量名称NewVar 1 用吸管吸取关
  • springboot+poi开发excel导出 加载Excel模板导出 Excel导出详解

    提到Excel导出功能 可能很多人都使用springmvc框架做过 笔者今天要给大家分享的是基于springBoot开发Excel复杂模板导出功能 所谓复杂模板指在模板里的特定表头里有不同的单元格合并以及背景色 字体颜色的填充 文本内容的对
  • linux下libpcap抓包分析

    linux下libpcap抓包分析 一 首先下载libpcap包http www tcpdump org latest release 然后安装 安装完成后进入安装根目录的tests文件夹 编译运行findalldevstest c 编译时
  • 实现mnist手写数字识别(第一周)

    本文为 365天深度学习训练营 中的学习记录博客 参考文章 365天深度学习训练营 第P1周 实现mnist手写数字识别 Pytorch实战 第P1周 实现mnist手写数字识别 qq com 原作者 K同学啊 接辅导 项目定制 我的环境
  • 使用codestriker搭建代码评审平台

    codestriker是用perl语言开发的 可以使用apache cgi进行访问的代码评审web站点 搭建过程如下 1 yum install perl 2 yum install highlight 3 配置codestriker co
  • k8s部署minio

    安装krew插件 官网地址 https krew sigs k8s io docs user guide setup install set x cd mktemp d OS uname tr upper lower
  • 图论算法<三>:判断有向图中是否有存在循环 ,以及环的个数和各个环中的元素

    1 目的 判断有向图中是否有存在循环 以及环的个数和各个环中的元素 2 示例效果 2 1 原始数据 路线起终点整理如下 共计12个顶点 19条边 起点 终点 1 最后的1代表起点终点是连通的 起点 终点 1 2 4 1 起点 终点 1 9
  • 深度学习理论及运用(三)Deep feedforward network

    转化成非线性 1 RBF径向基函数 2 利用sift特征以及K means 3 用数据去训练 举例 异或门 使用非线性运算 用图来表示实现线性可分的过程 目的 优化目标函数 Cost Functions 目标函数 交叉熵 公式前加负号转化为
  • Mybatis-Plus&&Druid多数据源配置

    多数据源配置思路 yml中配置多个数据源 通过AOP自动切换不同的数据源 配合Mybatis plus使用 yml配置 spring datasource druid db1 url jdbc mysql 10 168 1 118 3306