使用JdbcTemplate和dynamic-datasource-spring-boot-starter动态切换数据源操作数据库
记录:414
场景:使用JdbcTemplate和dynamic-datasource-spring-boot-starter动态切换数据源操作数据库。
版本:JDK 1.8,Spring Boot 2.6.3,dynamic-datasource-spring-boot-starter-3.3.2。
源码:https://github.com/baomidou/dynamic-datasource-spring-boot-starter
dynamic-datasource-spring-boot-starter:一个基于springboot的快速集成多数据源的启动器。
1.动态数据源注解@DS作用在类上
1.1类GetDataDao
@DS("hub_a_db")
@Repository
public class GetDataDao {@Autowiredprivate JdbcTemplate jt;public List<Map<String, Object>> getData() {String selectSQL = "SELECT CITY_ID,CITY_NAME,LAND_AREA,POPULATION, " +" GROSS,CITY_DESCRIBE,DATA_YEAR,UPDATE_TIME " +"FROM t_city ";List<Map<String, Object>> data = jt.queryForList(selectSQL);return data;}
}
1.2类InsertDataDao
@DS("hub_b_db")
@Repository
public class InsertDataDao {@Autowiredprivate JdbcTemplate jt;public void insertData(List<Map<String, Object>> data) {String insertSQL = "INSERT INTO t_city (\\n" +" CITY_ID,CITY_NAME,LAND_AREA,POPULATION,\\n" +" GROSS,CITY_DESCRIBE,DATA_YEAR,UPDATE_TIME)\\n" +"VALUES (?, ?, ?, ?, ?, ?, ?, ?)";jt.batchUpdate(insertSQL, new BatchPreparedStatementSetter() {@Overridepublic void setValues(PreparedStatement ps, int i) throws SQLException {Map<String, Object> oneRow = data.get(i);ps.setObject(1, oneRow.get("CITY_ID"));ps.setObject(2, oneRow.get("CITY_NAME"));ps.setObject(3, oneRow.get("LAND_AREA"));ps.setObject(4, oneRow.get("POPULATION"));ps.setObject(5, oneRow.get("GROSS"));ps.setObject(6, oneRow.get("CITY_DESCRIBE"));ps.setObject(7, oneRow.get("DATA_YEAR"));ps.setObject(8, oneRow.get("UPDATE_TIME"));}@Overridepublic int getBatchSize() {return data.size();}});}
}
2.动态数据源注解@DS作用在方法
@Repository
public class GetAndInsertDataDao {@Autowiredprivate JdbcTemplate jt;@DS("hub_a_db")public List<Map<String, Object>> getData() {String selectSQL = "SELECT CITY_ID,CITY_NAME,LAND_AREA,POPULATION, " +" GROSS,CITY_DESCRIBE,DATA_YEAR,UPDATE_TIME " +"FROM t_city ";List<Map<String, Object>> data = jt.queryForList(selectSQL);return data;}@DS("hub_b_db")public void insertData(List<Map<String, Object>> data) {String insertSQL = "INSERT INTO t_city (\\n" +" CITY_ID,CITY_NAME,LAND_AREA,POPULATION,\\n" +" GROSS,CITY_DESCRIBE,DATA_YEAR,UPDATE_TIME)\\n" +"VALUES (?, ?, ?, ?, ?, ?, ?, ?)";jt.batchUpdate(insertSQL, new BatchPreparedStatementSetter() {@Overridepublic void setValues(PreparedStatement ps, int i) throws SQLException {Map<String, Object> oneRow = data.get(i);ps.setObject(1, oneRow.get("CITY_ID"));ps.setObject(2, oneRow.get("CITY_NAME"));ps.setObject(3, oneRow.get("LAND_AREA"));ps.setObject(4, oneRow.get("POPULATION"));ps.setObject(5, oneRow.get("GROSS"));ps.setObject(6, oneRow.get("CITY_DESCRIBE"));ps.setObject(7, oneRow.get("DATA_YEAR"));ps.setObject(8, oneRow.get("UPDATE_TIME"));}@Overridepublic int getBatchSize() {return data.size();}});}
}
3.使用DynamicDataSourceContextHolder操作动态数据源
无注解,在调用时,使用DynamicDataSourceContextHolder操作动态数据源。
@Repository
public class GetAndInsertDataByHolderDao {@Autowiredprivate JdbcTemplate jt;public List<Map<String, Object>> getData() {String selectSQL = "SELECT CITY_ID,CITY_NAME,LAND_AREA,POPULATION, " +" GROSS,CITY_DESCRIBE,DATA_YEAR,UPDATE_TIME " +"FROM t_city ";List<Map<String, Object>> data = jt.queryForList(selectSQL);return data;}public void insertData(List<Map<String, Object>> data) {String insertSQL = "INSERT INTO t_city (\\n" +" CITY_ID,CITY_NAME,LAND_AREA,POPULATION,\\n" +" GROSS,CITY_DESCRIBE,DATA_YEAR,UPDATE_TIME)\\n" +"VALUES (?, ?, ?, ?, ?, ?, ?, ?)";jt.batchUpdate(insertSQL, new BatchPreparedStatementSetter() {@Overridepublic void setValues(PreparedStatement ps, int i) throws SQLException {Map<String, Object> oneRow = data.get(i);ps.setObject(1, oneRow.get("CITY_ID"));ps.setObject(2, oneRow.get("CITY_NAME"));ps.setObject(3, oneRow.get("LAND_AREA"));ps.setObject(4, oneRow.get("POPULATION"));ps.setObject(5, oneRow.get("GROSS"));ps.setObject(6, oneRow.get("CITY_DESCRIBE"));ps.setObject(7, oneRow.get("DATA_YEAR"));ps.setObject(8, oneRow.get("UPDATE_TIME"));}@Overridepublic int getBatchSize() {return data.size();}});}
}
4.测试类
4.1测试类
@Slf4j
@RestController
@RequestMapping("/hub/example/load01")
public class LoadController {@Autowiredprivate GetDataDao getDataDao;@Autowiredprivate InsertDataDao insertDataDao;@Autowiredprivate GetAndInsertDataDao getAndInsertDataDao;@Autowiredprivate GetAndInsertDataByHolderDao getAndInsertDataByHolderDao;/*** 1.动态数据源注解@DS作用在类上* */@GetMapping("/load01")public Object load01() {log.info("测试开始...");List<Map<String, Object>> data = getDataDao.getData();insertDataDao.insertData(data);log.info("测试结束...");return "执行成功";}/*** 2.动态数据源注解@DS作用在方法上* */@GetMapping("/load02")public Object load02() {log.info("测试开始...");List<Map<String, Object>> data = getAndInsertDataDao.getData();getAndInsertDataDao.insertData(data);log.info("测试结束...");return "执行成功";}/*** 3.使用DynamicDataSourceContextHolder操作动态数据源* */@GetMapping("/load03")public Object load03() {log.info("测试开始...");//1.使用hub_a_db数据源读数据DynamicDataSourceContextHolder.push("hub_a_db");List<Map<String, Object>> data = getAndInsertDataByHolderDao.getData();//2.使用hub_b_db数据源写数据DynamicDataSourceContextHolder.poll();DynamicDataSourceContextHolder.push("hub_b_db");getAndInsertDataByHolderDao.insertData(data);log.info("测试结束...");return "执行成功";}
}
4.2测试URL
URL01: http://127.0.0.1:18204/hub-example/hub/example/load01/load01
URL02: http://127.0.0.1:18204/hub-example/hub/example/load01/load02
URL03: http://127.0.0.1:18204/hub-example/hub/example/load01/load03
5.基础配置
5.1配置动态数据源
spring:jackson:time-zone: GMT+8datasource:dynamic:primary: hub_a_dbstrict: falsedatasource:hub_a_db:url: jdbc:mysql://127.0.0.1:3306/hub_a_dbusername: hub_apassword: 12345678driver-class-name: com.mysql.cj.jdbc.Driverhub_b_db:url: jdbc:mysql://127.0.0.1:3306/hub_b_dbusername: hub_bpassword: 12345678driver-class-name: com.mysql.cj.jdbc.Driver
5.2动态数据源依赖包
<dependency><groupId>com.baomidou</groupId><artifactId>dynamic-datasource-spring-boot-starter</artifactId><version>3.3.2</version>
</dependency>
5.3建表语句
CREATE TABLE t_city (CITY_ID BIGINT(16) NOT NULL COMMENT '唯一标识',CITY_NAME VARCHAR(64) COLLATE utf8_bin NOT NULL COMMENT '城市名',LAND_AREA DOUBLE DEFAULT NULL COMMENT '城市面积',POPULATION BIGINT(16) DEFAULT NULL COMMENT '城市人口',GROSS DOUBLE DEFAULT NULL COMMENT '生产总值',CITY_DESCRIBE VARCHAR(512) COLLATE utf8_bin DEFAULT NULL COMMENT '城市描述',DATA_YEAR VARCHAR(16) COLLATE utf8_bin DEFAULT NULL COMMENT '数据年份',UPDATE_TIME DATETIME DEFAULT NULL COMMENT '更新时间'
) ENGINE=INNODB DEFAULT CHARSET=utf8 COLLATE=utf8_bin COMMENT='城市信息表';
以上,感谢。
2023年4月17日