本例是在【Mybatis】MyBatis之配置多数据源(十)的基础上进行拓展,查看本例请先学习第十章

实现原理

  1、扩展Spring的AbstractRoutingDataSource抽象类(该类充当了DataSource的路由中介, 能有在运行时, 根据某种key值来动态切换到真正的DataSource上。)

     从AbstractRoutingDataSource的源码中,有一个数据源查找属性(dataSourceLookup),和一个 查询数据源方法 (resolveSpecifiedDataSource):

 private DataSourceLookup dataSourceLookup = new JndiDataSourceLookup();

 protected DataSource resolveSpecifiedDataSource(Object dataSource) throws IllegalArgumentException {
if (dataSource instanceof DataSource) {
return (DataSource) dataSource;
}
else if (dataSource instanceof String) {
return this.dataSourceLookup.getDataSource((String) dataSource);
}
else {
throw new IllegalArgumentException(
"Illegal data source value - only [javax.sql.DataSource] and String supported: " + dataSource);
}
}

  2、类中数据源查找类使用的是JndiDataSourceLookup,此类查找的是JNDI数据源,参考:【Tomcat】Tomcat 配置JNDI数据源(三)

  3、我们可以自定义动态数据源查找类来,来控制自定义数据源

实现案例

  1、搭建项目,参考:【Mybatis】MyBatis之配置多数据源(十)

  2、自定义数据源查找类

 package com.test.datasource;

 import java.beans.PropertyVetoException;
import java.sql.SQLException;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; import javax.sql.DataSource; import org.springframework.jdbc.datasource.lookup.DataSourceLookup;
import org.springframework.jdbc.datasource.lookup.DataSourceLookupFailureException;
import org.springframework.util.Assert; import com.mchange.v2.c3p0.ComboPooledDataSource; public class DynamicDataSourceLookup implements DataSourceLookup { /**
* 数据源集合
*/
private final Map<String, DataSource> dataSources = new ConcurrentHashMap<>(); /**
* 根据数据源名称,获取数据源
*/
@Override
public DataSource getDataSource(String dataSourceName) throws DataSourceLookupFailureException { Assert.notNull(dataSourceName, "DataSourceName must not be null");
DataSource dataSource = this.dataSources.get(dataSourceName);
if (dataSource == null) {
// datasource not in map.. create one and add to map
dataSource = createDataSource(dataSourceName);
if(dataSource != null) {
addDataSource(dataSourceName, dataSource);
}
}
// 记录数据源状态
if(dataSource instanceof ComboPooledDataSource) {
ComboPooledDataSource ds = (ComboPooledDataSource) dataSource;
try {
System.out.println("ds.getMaxPoolSize():" + ds.getMaxPoolSize());// 最大连接数
System.out.println("ds.getMaxPoolSize():" + ds.getMaxPoolSize());// 最大连接数
System.out.println("ds.getMinPoolSize():" + ds.getMinPoolSize());// 最小连接数
System.out.println("ds.getNumBusyConnections():" + ds.getNumBusyConnections());// 正在使用连接数
System.out.println("ds.getNumIdleConnections():" + ds.getNumIdleConnections());// 空闲连接数
System.out.println("ds.getNumConnections():" + ds.getNumConnections());
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}// 总连接数
}
return dataSource;
} /**
* 创建数据源
* @param dataSourceName
* @return
*/
private DataSource createDataSource(String dataSourceName) { ComboPooledDataSource dataSource = null; // 可根据其他业务数据,创建数据源
try {
dataSource = new ComboPooledDataSource();
dataSource.setUser("hd");
dataSource.setPassword("hd123456");
dataSource.setJdbcUrl("jdbc:mysql://mysql.naughty7878.top:3306/test_mybatis?allowPublicKeyRetrieval=true");
dataSource.setDriverClass("com.mysql.jdbc.Driver");
dataSource.setDataSourceName(dataSourceName); } catch (PropertyVetoException e) {
throw new RuntimeException("Error creating data source");
}
return dataSource;
} /**
* 添加数据源到 数据源集合中
* @param dataSourceName
* @param dataSource
*/
public void addDataSource(String dataSourceName, DataSource dataSource) {
Assert.notNull(dataSourceName, "DataSourceName must not be null");
Assert.notNull(dataSource, "DataSource must not be null");
this.dataSources.put(dataSourceName, dataSource);
} /**
* 从数据源集合移除数据源
* @param dataSourceName
*/
public void removeDataSource(String dataSourceName) {
Assert.notNull(dataSourceName, "DataSourceName must not be null");
this.dataSources.remove(dataSourceName);
} }

  3、编辑动态数据源类,从写determineTargetDataSource方法,查询数据源时,先从自定义数据源中查找,然后从内部数据源中查找

 package com.test.datasource;

 import javax.sql.DataSource;

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

 /**
* 动态数据源(依赖于spring)
* @author chenheng
* @date 2019-08-03 17:27:35
*
*/
public class DynamicDataSource extends AbstractRoutingDataSource { @Override
protected Object determineCurrentLookupKey() {
return DataSourceHolder.getDataSourceKey();
} @Override
protected DataSource determineTargetDataSource() { Object lookupKey = determineCurrentLookupKey();
DataSource dataSource = null;
// 自定义外部数据源类中,查询数据源
if(lookupKey != null) {
dataSource = resolveSpecifiedDataSource(key);
}
// 在内部数据源中,查询数据源
if(dataSource == null){
dataSource= super.determineTargetDataSource();
}
return dataSource;
}
}

  4、注入dynamicDataSourceLookup,并且在dataSource注入属性dataSourceLookup

 <bean id="dynamicDataSourceLookup" class="com.test.datasource.DynamicDataSourceLookup">
</bean> <!-- 数据源:Spring用来控制业务逻辑。数据源、事务控制、aop -->
<bean id="dataSource" class="com.test.datasource.DynamicDataSource">
<property name="targetDataSources">
<map key-type="java.lang.String">
<entry key="dataSource1" value-ref="dataSource1"></entry>
<entry key="dataSource2" value-ref="dataSource2"></entry>
</map>
</property>
<!-- 默认目标数据源为你主库数据源 -->
<property name="defaultTargetDataSource" ref="dataSource1"/>
<property name="dataSourceLookup" ref="dynamicDataSourceLookup"/>
</bean>

  5、编辑是一个动态数据源DAO,来使用自定动态数据源

 package com.test.dao;

 import javax.sql.DataSource;

 import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
import org.springframework.stereotype.Repository; @Repository
public class DynamicDao extends JdbcDaoSupport { @Autowired
public DynamicDao(DataSource dataSource) {
this.setDataSource(dataSource);
} }

  6、在Service中使用

 package com.test.service;

 import java.util.List;

 import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.stereotype.Service; import com.test.dao.DynamicDao;
import com.test.datasource.DataSourceHolder;
import com.test.pojo.Employee; @Service
public class DynamicService { @Autowired
DynamicDao dynamicDao; public List<Employee> getEmployee() {
DataSourceHolder.setDataSourceKey("自定义");
return dynamicDao.getJdbcTemplate().query("select id, last_name lastName, gender, email from employee",
new BeanPropertyRowMapper<>(Employee.class));
} }

数据流转顺序:

  1.在 DataSourceHolder中,设置数据源名称

  2.Spring 调用determineCurrentLookupKey()方法<DynamicDataSource中重写AbstractRoutingDataSource类中的方法> ,从DataSourceHolder取出当前的数据库名称,并返回

  3.AbstractRoutingDataSource类中determineTargetDataSource()方法调用determineCurrentLookupKey(),先匹配外部数据库,然后匹配内部数据库;

  4.匹配到指定的数据库,并建立链接,即为切换到相应的数据库;

  5.在指定的数据库中执行相应的sql

【Mybatis】MyBatis之配置自定义数据源(十一)的更多相关文章

  1. springboot+mybatis +yml文件配置多数据源

    记录一下java开发中多数据源的配置过程, 参考博客:https://blog.csdn.net/weinichendian/article/details/72903757,我在这里进行了整理,使用 ...

  2. mybatis数据库基本配置包括数据源事物类型等

    <?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE configuration PUBLIC ...

  3. [教程] Spring+Mybatis环境配置多数据源

    一.简要概述 在做项目的时候遇到需要从两个数据源获取数据,项目使用的Spring + Mybatis环境,看到网上有一些关于多数据源的配置,自己也整理学习一下,然后自动切换实现从不同的数据源获取数据功 ...

  4. Spring Boot + MyBatis + Pagehelper 配置多数据源

    前言: 本文为springboot结合mybatis配置多数据源,在项目当中很多情况是使用主从数据源来读写分离,还有就是操作多库,本文介绍如何一个项目同时使用2个数据源. 也希望大家带着思考去学习!博 ...

  5. 【Mybatis】MyBatis之配置多数据源(十)

    在做项目的过程中,有时候一个数据源是不够,那么就需要配置多个数据源.本例介绍mybatis多数据源配置 前言 一般项目单数据源,使用流程如下: 单个数据源绑定给sessionFactory,再在Dao ...

  6. springboot配置多数据源mybatis配置失效问题

    mybatis配置 #开启驼峰映射 mybatis.configuration.map-underscore-to-camel-case=true #开启打印sql mybatis.configura ...

  7. mybatis源码解读(三)——数据源的配置

    在mybatis-configuration.xml 文件中,我们进行了如下的配置: <!-- 可以配置多个运行环境,但是每个 SqlSessionFactory 实例只能选择一个运行环境常用: ...

  8. spring+myBatis 配置多数据源,切换数据源

    注:本文来源于  tianzhiwuqis <spring+myBatis 配置多数据源,切换数据源> 一个项目里一般情况下只会使用到一个数据库,但有的需求是要显示其他数据库的内容,像这样 ...

  9. SpringBoot系列七:SpringBoot 整合 MyBatis(配置 druid 数据源、配置 MyBatis、事务控制、druid 监控)

    1.概念:SpringBoot 整合 MyBatis 2.背景 SpringBoot 得到最终效果是一个简化到极致的 WEB 开发,但是只要牵扯到 WEB 开发,就绝对不可能缺少数据层操作,所有的开发 ...

随机推荐

  1. NETTY框架的使用

    一.Netty 简介 Netty 是基于 Java NIO 的异步事件驱动的网络应用框架,使用 Netty 可以快速开发网络应用,Netty 提供了高层次的抽象来简化 TCP 和 UDP 服务器的编程 ...

  2. vue引入js文件时报This dependency was not found:错误

    vue引入js文件时报This dependency was not found:错误 使用了很多方法,原来是这么小的问题,特此记录 解决办法 添加 ./

  3. JS判断移动端访问设备并加载对应CSS样式

    JS判断不同web访问环境,主要针对移动设备,提供相对应的解析方案(判断设备代码直接copy腾讯网的) // 判断是否为移动端运行环境 if(/AppleWebKit.*Mobile/i.test(n ...

  4. 织梦dedecms会员中心分类管理无法修改、删除分类名

    member/mtypes.PHP 文件中添加 另外,member/myfriend_group.php文件中也存在同样的问题,也要添加,不添加的话好友分组中也是同样问题

  5. PAT乙级1045 快速排序

    1045 快速排序 (25分)   著名的快速排序算法里有一个经典的划分过程:我们通常采用某种方法取一个元素作为主元,通过交换,把比主元小的元素放到它的左边,比主元大的元素放到它的右边. 给定划分后的 ...

  6. 学到了林海峰,武沛齐讲的Day37 完

    day1   多用户同时刻下载上传程序分析 day2   htlm介绍 觉得收货的季节到了 day3   htlm介绍 day4   htlm介绍 关键字介绍  Toray大仙 Toray大仙 day ...

  7. PHP 高手博客网站集合

    风雪之隅-Laruence的博客 韩天峰(Rango)的博客 我的志愿是做一个校长 张宴的博客 - Web系统架构与底层研发 沈逸的个人站点 博学无忧 - 信海龙的博客

  8. noi.ac #42 模拟

    \(des\) 二维平面上存在 \(m\) 个点,每个点会对该点的 \(8\) 个方向上的最近的点产生影响 问每个点会被影响多少次 \(sol\) 过每个点会产生 \(4\) 条线段 保存每条线段的斜 ...

  9. Cogs 1708. 斐波那契平方和(矩阵乘法)

    斐波那契平方和 ★★☆ 输入文件:fibsqr.in 输出文件:fibsqr.out 简单对比 时间限制:0.5 s 内存限制:128 MB [题目描述] ,对 1000000007 取模.F0=0, ...

  10. Angular惰性加载的特性模块

    一:Angular-CLI建立应用 cmd命令:ng new lazy-app --routing    (创建一个名叫 lazy-app 的应用,而 --routing 标识生成了一个名叫 app- ...