【sping揭秘】21、Spring动态数据源的切换
对于多个数据源的时候,我们如何切换不同的数据源进行数据库的操作呢?
当然我们可以直接定义2个DataSource,然后在每次获取connection的时候,从不同的DataSource中获取connection,类似如下
这种情况可以是2个数据库存放的数据性质是不同的,DataSource1存放1种数据,DataSource2存放另一种数据,每个数据库承担不同的数据访问请求,这2个是完全相互独立不相干的
这种就比较简单,那就是直接定义不同的jdbctemplate,设置不同的DataSource就可以了,但是要注意代码编写的时候入库操作
还有一种就是数据性质是一样的,不同的数据源失去了独立自主的地位,这样所有的数据访问我们需要通过“盟主”进行。
这里我们可以借助spring的AbstractRoutingDataSource进行分发
后面几个数据库之间的数据共享,我们可以进行数据库数据的主从复制
定义数据源
package cn.cutter.start.database; import javax.sql.DataSource; import org.apache.commons.dbcp2.BasicDataSource;
import org.springframework.beans.factory.FactoryBean;
import org.springframework.stereotype.Component; /**
* 自定义DataSource
* @author xiaof
*
*/
@Component
public class LiferayDataSource1 implements FactoryBean<DataSource> { @Override
public DataSource getObject() throws Exception {
BasicDataSource dataSource = new BasicDataSource(); //设置相应的参数
//1、数据库驱动类
dataSource.setDriverClassName("com.mysql.jdbc.Driver");
//2、url,用户名,密码
dataSource.setUrl("jdbc:mysql://localhost:3306/liferay?characterEncoding=utf-8");
dataSource.setUsername("liferay"); dataSource.setPassword("xiaofeng2017");
//3、初始化连接大小
dataSource.setInitialSize(1);
//4、连接池最大数据量
dataSource.setMaxTotal(500);
//5、连接池最大小空闲
dataSource.setMinIdle(1);
dataSource.setMaxIdle(20);
//6、最大等待时间 单位毫秒
dataSource.setMaxWaitMillis(20 * 1000);
//7、指明连接是否被空闲连接回收器(如果有)进行检验
dataSource.setPoolPreparedStatements(true);
//8、运行一次空闲连接回收器的时间间隔(60秒)
dataSource.setTimeBetweenEvictionRunsMillis(60 * 1000);
//9、验证时使用的SQL语句
dataSource.setValidationQuery("SELECT 1 FROM DUAL");
//10、借出连接时不要测试,否则很影响性能
//11、申请连接的时候检测,如果空闲时间大于 timeBetweenEvictionRunsMillis,执行validationQuery检测连接是否有效
dataSource.setTestWhileIdle(false); return dataSource;
} @Override
public Class<?> getObjectType() {
// TODO Auto-generated method stub
return DataSource.class;
} }
package cn.cutter.start.database; import javax.sql.DataSource; import org.apache.commons.dbcp2.BasicDataSource;
import org.springframework.beans.factory.FactoryBean;
import org.springframework.stereotype.Component; @Component
public class LiferayDataSource2 implements FactoryBean<DataSource> { @Override
public DataSource getObject() throws Exception {
BasicDataSource dataSource = new BasicDataSource(); //设置相应的参数
//1、数据库驱动类
dataSource.setDriverClassName("com.mysql.jdbc.Driver");
//2、url,用户名,密码
dataSource.setUrl("jdbc:mysql://localhost:3306/test?characterEncoding=utf-8");
dataSource.setUsername("liferay"); dataSource.setPassword("xiaofeng2017");
//3、初始化连接大小
dataSource.setInitialSize(1);
//4、连接池最大数据量
dataSource.setMaxTotal(500);
//5、连接池最大小空闲
dataSource.setMinIdle(1);
dataSource.setMaxIdle(20);
//6、最大等待时间 单位毫秒
dataSource.setMaxWaitMillis(20 * 1000);
//7、指明连接是否被空闲连接回收器(如果有)进行检验
dataSource.setPoolPreparedStatements(true);
//8、运行一次空闲连接回收器的时间间隔(60秒)
dataSource.setTimeBetweenEvictionRunsMillis(60 * 1000);
//9、验证时使用的SQL语句
dataSource.setValidationQuery("SELECT 1 FROM DUAL");
//10、借出连接时不要测试,否则很影响性能
//11、申请连接的时候检测,如果空闲时间大于 timeBetweenEvictionRunsMillis,执行validationQuery检测连接是否有效
dataSource.setTestWhileIdle(false); return dataSource;
} @Override
public Class<?> getObjectType() {
// TODO Auto-generated method stub
return DataSource.class;
} }
package cn.cutter.start.database; import java.util.HashMap; import javax.annotation.PostConstruct;
import javax.sql.DataSource; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Component; /**
* 集合所有的数据源
* @author xiaof
*
*/
@Component
public class DataSources extends HashMap<Integer, DataSource> { @Autowired
@Qualifier("liferayDataSource1")
private DataSource liferayDataSource1; @Autowired
@Qualifier("liferayDataSource2")
private DataSource liferayDataSource2; @PostConstruct //创建对象之前进行注入
public void initDataSource() {
this.put(0, liferayDataSource1);
this.put(1, liferayDataSource2);
} }
动态路由类设置
package cn.cutter.start.database; import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; import javax.annotation.Resource; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
import org.springframework.stereotype.Component; /**
* spring 中提供多数据源 高可用支持,合纵连横 多数据源
* @author xiaof
*
*/
@Component("multipleDataSource")
public class MultipleDataSource extends AbstractRoutingDataSource { private static final Log logger = LogFactory.getLog(MultipleDataSource.class); private Lock lock = new ReentrantLock();
private int counter = 0;
private int dataSourceNumber = 2; //对这个类的继承过来的对象进行分化
//1、注入defaultTargetDataSource
@Resource(name="liferayDataSource1")
public void setDefaultTargetDataSource(Object defaultTargetDataSource) {
// 对于父类私有成员,就只能通过super访问
super.setDefaultTargetDataSource(defaultTargetDataSource);
} //2、注入所有数据源对象map targetDataSources
@Resource(name="dataSources")
@Override
public void setTargetDataSources(Map<Object, Object> targetDataSources) {
super.setTargetDataSources(targetDataSources);
} @Override
protected Object determineCurrentLookupKey() { lock.lock(); try { ++counter;
//根据取余来进行取数对应的数据库源, 可以自定义自己的规则 来判断如何切换数据库
int lookupKey = counter % getDataSourceNumber();
logger.info("获取第:" + lookupKey + " 数据源");
return new Integer(lookupKey); } finally {
lock.unlock();
}
} public Lock getLock() {
return lock;
} public void setLock(Lock lock) {
this.lock = lock;
} public int getCounter() {
return counter;
} public void setCounter(int counter) {
this.counter = counter;
} public int getDataSourceNumber() {
return dataSourceNumber;
} public void setDataSourceNumber(int dataSourceNumber) {
this.dataSourceNumber = dataSourceNumber;
}
}
最后设置我们的动态jdbctemplate
package cn.cutter.start.database; import org.springframework.beans.factory.FactoryBean;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Component; /**
* 多数据源切换,jdbctemplate使用
* @author xiaof
*
*/
@Component
public class MultipleJdbcTemplate implements FactoryBean<JdbcTemplate>{ @Autowired
@Qualifier("multipleDataSource")
private MultipleDataSource multipleDataSource; @Override
public JdbcTemplate getObject() throws Exception { JdbcTemplate jdbcTemplate = new JdbcTemplate(multipleDataSource);
return jdbcTemplate;
} @Override
public Class<?> getObjectType() {
// TODO Auto-generated method stub
return JdbcTemplate.class;
} }
读取数据
package spring.vo; public class DataVo {
private int num;
private String name = "cutter_point";
public int getNum() {
return num;
}
public void setNum(int num) {
this.num = num;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
} }
测试
@Test
public void testMultipleDataSource() {
ApplicationContext ctx = this.before(); //循环向数据库插入数据
String sql = "insert into multipleDataSourceTestTable values (?, ?)"; for(int i = 0; i < 10; ++i) {
//获取相应的数据连接 模拟项目中不同的业务场景获取jdbctemplate对象
JdbcTemplate jdbcTemplate = (JdbcTemplate) ctx.getBean("multipleJdbcTemplate");
DataVo dataVo = new DataVo();
dataVo.setNum(i);
jdbcTemplate.update(sql, new PreparedStatementSetter() { @Override
public void setValues(PreparedStatement ps) throws SQLException {
ps.setInt(1, dataVo.getNum());
ps.setString(2, dataVo.getName());
}
});
} // List<DataVo> datas = new ArrayList<DataVo>();
// for(int i = 0; i < 10; ++i) {
// DataVo dataVo = new DataVo();
// dataVo.setNum(i);
// datas.add(dataVo);
// }
//
// jdbcTemplate.batchUpdate(sql, new BatchPreparedStatementSetter() {
//
// @Override
// public void setValues(PreparedStatement ps, int i) throws SQLException {
// DataVo dataVo = datas.get(i);
// ps.setInt(1, dataVo.getNum());
// ps.setString(2, dataVo.getName());
// }
//
// @Override
// public int getBatchSize() {
// return datas.size();
// }
// }); }
【sping揭秘】21、Spring动态数据源的切换的更多相关文章
- Spring动态数据源的配置
Spring动态数据源 我们很多项目中业务都需要涉及到多个数据源,就是对不同的方法或者不同的包使用不同的数据源.最简单的做法就是直接在Java代码里面lookup需要的数据源,但是这种做法耦合性太高, ...
- Spring动态数据源实现读写分离
一.创建基于ThreadLocal的动态数据源容器,保证数据源的线程安全性 package com.bounter.mybatis.extension; /** * 基于ThreadLocal实现的动 ...
- spring动态数据源+事务
今天在尝试配置spring的动态数据源和事务管理的时候,遇到了几处配置上的问题,在此记录下: 1.使用了spring的aop思想,实现了动态数据源的切换. 2.spring的事务管理,是基于数据源的, ...
- Spring动态数据源-AbstractRoutingDataSource
在分库分表的情况下,在执行SQL时选择连接不同的数据源(库)的思路:配置多个数据源加到动态数据源对象中,根据实际的情况动态切换到相应的数据源中. 如存放订单信息的有10个库,每个库中有100张表,根据 ...
- Spring 动态创建并切换数据源
公司要求后端项目可以进行动态创建并切换数据源,看了网上很多例子大多数使用的都是Spring内置的AbstractRoutingDataSource进行的,使用此方法不是不行但是有诸多缺陷,比如切换时需 ...
- spring 动态数据源
1.动态数据源: 在一个项目中,有时候需要用到多个数据库,比如读写分离,数据库的分布式存储等等,这时我们要在项目中配置多个数据库. 2.原理: (1).spring 单数据源获取数据连接过程: ...
- @Transactional导致AbstractRoutingDataSource动态数据源无法切换的解决办法
上午花了大半天排查一个多数据源主从切换的问题,记录一下: 背景: 项目的数据库采用了读写分离多数据源,采用AOP进行拦截,利用ThreadLocal及AbstractRoutingDataSource ...
- Spring Boot数据访问之动态数据源切换之使用注解式AOP优化
在Spring Boot数据访问之多数据源配置及数据源动态切换 - 池塘里洗澡的鸭子 - 博客园 (cnblogs.com)中详述了如何配置多数据源及多数据源之间的动态切换.但是需要读数据库的地方,就 ...
- Spring Boot + Mybatis 实现动态数据源
动态数据源 在很多具体应用场景的时候,我们需要用到动态数据源的情况,比如多租户的场景,系统登录时需要根据用户信息切换到用户对应的数据库.又比如业务A要访问A数据库,业务B要访问B数据库等,都可以使用动 ...
随机推荐
- 初学c# -- 开始学directx
这些天对directx有兴趣了,开始慢慢学,先学基础,找了好些资料,为毛都写的辣么长呢,学习精简下来就几行. 安装个directx sdk,在win10里面文件夹C:\Windows\Microsof ...
- mysql case when then else end 写法
今天改SQL用到了就搜了搜,现在记下来. case when tableName.type in (1,2,3,4) and tableName.state in (5,6) then 1 when ...
- poj2115(扩展欧基里德定理)
题目链接:https://vjudge.net/problem/POJ-2115 题意:模拟for循环for(int i=A;i!=B;i+=C),且数据范围为k位无符号数以内,即0~1<< ...
- 微擎开发------day02
基本要素: 微擎系统数据库操作使用PDO兼容方式,参数绑定进行查询操作 (1) 数据表加上表前缀 $sql = "select * from ".tablename('users ...
- linux学习第十二天 (Linux就该这么学)找到一本不错的Linux电子书,附《Linux就该这么学》章节目录
本书是由全国多名红帽架构师(RHCA)基于最新Linux系统共同编写的高质量Linux技术自学教程,极其适合用于Linux技术入门教程或讲课辅助教材,目前是国内最值得去读的Linux教材,也是最有价值 ...
- JVM 字节码(二)方法表详解
JVM 字节码(二)方法表和属性表 上一节中对 ClassFile 的整体进行了五个详细的说明, 本节围绕 ClassFile 最重要的一个内容 - 方法表的 Code 属性展开 ,更多 JVM Me ...
- Oracle 触发器和序列的创建和使用 (自动增长列)
-- 创建序列 create sequence 序列名称 start with 1 -- 起始值 increment by 1 -- 增量 maxvalue ...
- AX_Currency
Currency::curAmount(9.23,"HKD"); Currency::curAmount2CurAmount(9.23,"RMB"," ...
- 【ElasticSearch】 安装
Elasticsearch简介 Elasticsearch 是一个开源的分布式 RESTful 搜索和分析引擎,能够解决越来越多不同的应用场景 官网地址:https://www.elastic.co ...
- kbmmw 5.07 正式发布
来了来了 5.07.00 Dec 9 2018 Important notes (changes that may break existing code) === ...