springboot+druid+mybatis plus的多数据源配置

记得面试时候,有面试官会问道,你们多数据源是怎么实现的呀。.......,一阵蒙蔽中,然后说道我们之前项目中,没有用到多数据源。

所幸,目前做得项目中有一个业务逻辑中,用到多个数据库数据情况,多数据源华丽上线。

一. mybatis plus

 因为我们项目是springboot+mybatis plus,有些人一看,mybatis还知道对吧,mybatis plus是什么鬼,其实字面意思可以理解,就是对mybatis进行一些功能改造,一些封装升级,然后用起来特别方便。

     核心功能的升级主要是以下三点:

     支持通用的 CRUD、代码生成器与条件构造器。

通用 CRUD:定义好 Mapper 接口后,只需要继承 BaseMapper 接口即可获得通用的增删改查功能,无需编写任何接口方法与配置文件

条件构造器:通过 EntityWrapper (实体包装类),可以用于拼接 SQL 语句,并且支持排序、分组查询等复杂的 SQL

代码生成器:支持一系列的策略配置与全局配置,比 MyBatis 的代码生成更好用

二.多数据源配置开始

    思路:

  1、yml中配置多个数据源信息

  2、通过AOP切换不同数据源

  3、配合mybatis plus使用


1、yml配置

spring:

  aop:

      proxy-target-class: true

      auto: true

  datasource:

    druid:

      db1:

        url: jdbc:mysql://localhost:3306/eboot

        username: root

        password: root

        driver-class-name: com.mysql.jdbc.Driver

        initialSize: 5

        minIdle: 5

        maxActive: 20

      db2:

        url: jdbc:oracle:thin:@192.168.136.222:ORCL

        username: sa

        password: sa123456

        driver-class-name: oracle.jdbc.OracleDriver

        initialSize: 5

        minIdle: 5

        maxActive: 20

2、启动多个数据源

@EnableTransactionManagement //开启事务

@Configuration  //spring中常用到注解,与xml配置相对立。是两种加载bean方式

@MapperScan("com.df.openapi.**.mapper.db*") // 扫描mapperdao的地址

public class MybatisPlusConfig {

    @Bean

    public PaginationInterceptor paginationInterceptor() {

        PaginationInterceptor paginationInterceptor = new PaginationInterceptor();

//        paginationInterceptor.setLocalPage(true); // 由于版本问题,有些类可能招不到这个方法,需要升级jar包

        return paginationInterceptor;

    }

    @Bean(name = "db1")

    @ConfigurationProperties(prefix = "spring.datasource.druid.db1")

    public DataSource db1() {

        return DruidDataSourceBuilder.create().build();

    }

    @Bean(name = "db2")

    @ConfigurationProperties(prefix = "spring.datasource.druid.db2")

    public DataSource db2() {

        return DruidDataSourceBuilder.create().build();

    }

    /**

    * 动态数据源配置

    *

    * @return

    */

    @Bean

    @Primary

    public DataSource multipleDataSource(@Qualifier("db1") DataSource db1,

                                        @Qualifier("db2") DataSource db2) {

        DynamicDataSource dynamicDataSource = new DynamicDataSource();

        Map<Object, Object> targetDataSources = new HashMap<>();

        targetDataSources.put(DBTypeEnum.db1.getValue(), db1);

        targetDataSources.put(DBTypeEnum.db2.getValue(), db2);

        dynamicDataSource.setTargetDataSources(targetDataSources);

        dynamicDataSource.setDefaultTargetDataSource(db2); // 程序默认数据源,这个要根据程序调用数据源频次,经常把常调用的数据源作为默认

        return dynamicDataSource;

    }

    @Bean("sqlSessionFactory")

    public SqlSessionFactory sqlSessionFactory() throws Exception {

        MybatisSqlSessionFactoryBean sqlSessionFactory = new MybatisSqlSessionFactoryBean();

        sqlSessionFactory.setDataSource(multipleDataSource(db1(), db2()));

        MybatisConfiguration configuration = new MybatisConfiguration();

        configuration.setJdbcTypeForNull(JdbcType.NULL);

        configuration.setMapUnderscoreToCamelCase(true);

        configuration.setCacheEnabled(false);

        sqlSessionFactory.setConfiguration(configuration);

        //PerformanceInterceptor(),OptimisticLockerInterceptor()

        //添加分页功能

        sqlSessionFactory.setPlugins(new Interceptor[]{

                paginationInterceptor()

        });

//        sqlSessionFactory.setGlobalConfig(globalConfiguration()); //注释掉全局配置,因为在xml中读取就是全局配置

        return sqlSessionFactory.getObject();

    }

/*  @Bean

    public GlobalConfiguration globalConfiguration() {

        GlobalConfiguration conf = new GlobalConfiguration(new LogicSqlInjector());

        conf.setLogicDeleteValue("-1");

        conf.setLogicNotDeleteValue("1");

        conf.setIdType(0);

        conf.setMetaObjectHandler(new MyMetaObjectHandler());

        conf.setDbColumnUnderline(true);

        conf.setRefresh(true);

        return conf;

    }*/

}

3、DBType枚举类

package com.df.openapi.config.db;

public enum DBTypeEnum {

    db1("db1"), db2("db2");

    private String value;

    DBTypeEnum(String value) {

        this.value = value;

    }

    public String getValue() {

        return value;

    }

}

4、动态数据源决策

package com.df.openapi.config.db;

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

public class DynamicDataSource extends AbstractRoutingDataSource {

private static final Logger LOGGER = LoggerFactory.getLogger(DynamicDataSource.class);

@Override

protected Object determineCurrentLookupKey() {

String datasource = DataSourceContextHolder.getDbType();

LOGGER.debug("使用数据源 {}", datasource);

return datasource;

}

}

5、设置、获取数据源

public class DataSourceContextHolder {

    private static final Logger LOGGER = LoggerFactory.getLogger(DataSourceContextHolder.class);

    private static final ThreadLocal contextHolder = new ThreadLocal<>(); //实际上就是开启多个线程,每个线程进行初始化一个数据源

    /**

    * 设置数据源

    * @param dbTypeEnum

    */

    public static void setDbType(DBTypeEnum dbTypeEnum) {

        contextHolder.set(dbTypeEnum.getValue());

    }

    /**

    * 取得当前数据源

    * @return

    */

    public static String getDbType() {

        return (String) contextHolder.get();

    }

    /**

    * 清除上下文数据

    */

    public static void clearDbType() {

        contextHolder.remove();

    }

}

6、AOP实现的数据源切换

@Order设置的足够小是为了让他先执行

/** * aop的实现的数据源切换

* aop切点,实现mapper类找寻,找到所属大本营以后,如db1Aspect(),则会调用

* db1()前面之前的操作,进行数据源的切换。 */@Component@Order(value = -100)@Slf4j@Aspectpublic class DataSourceAspect {    @Pointcut("execution(* com.zwyl.bazhong.dao.mapper.db1..*.*(..))")    private void db1Aspect() {    }    @Pointcut("execution(* com.zwyl.bazhong.dao.mapper.db2..*.*(..))")    private void db2Aspect() {    }    @Before("db1Aspect()")    public void db1() {        log.info("切换到db1 数据源...");        DataSourceContextHolder.setDbType(DBTypeEnum.db1);    }    @Before("db2Aspect()")    public void db2() {        log.info("切换到db2 数据源...");        DataSourceContextHolder.setDbType(DBTypeEnum.db2);    }}

7、mapper层结构

8、写一个service测试一下

@Service

public class DictServiceImpl implements IDictService {

    @Resource

    private PtDictMapper ptDictMapper; //来自db1

    @Resource

    private SysDictMapper sysDictMapper; // 来自db2

    @Override

    public void getById(String id) {

        PtDict dict = ptDictMapper.selectById("2bf6257fc8fe483c84c1ad7e89d632f6");

        SysDict sysDict = sysDictMapper.getById("49");

        System.out.println("123");

    }

}

9、测试结果

总结: 其实整个过程可以理解成,配置多数据源 xml中  -------> 然后通过加载多数源到spring工厂中-------->然后创建多线程,每个数据源对应一个数据源--------->然后实际调用时候,会先通过aop匹配到某一具体数据源------------->然后实例化当前数据源

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,902评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,037评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,978评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,867评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,763评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,104评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,565评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,236评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,379评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,313评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,363评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,034评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,637评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,719评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,952评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,371评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,948评论 2 341

推荐阅读更多精彩内容