SSM 最灵活实现动态切换操作多数据源

之前写过一篇极易上手的操作多数据源(https://blog.csdn.net/qq_35387940/article/details/99324108),基本看了就能整合实现多数据源了。
 

当前这篇文章,区别为,我们将会把多个数据源的信息放在一张数据库配置表 jdbc_config里面去,项目开始运行的时候,会从默认连接的数据库的这个配置表获取多数据源的信息,进行数据源的加载设置。  

OK,现在开始,首先我们把项目默认数据库写下,jdbc.properties:

jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://127.0.0.1:3306/game_message?useUnicode=true&characterEncoding=UTF-8&serverTimezone=GMT%2B8
jdbc.username=root
jdbc.password=root

jdbc.initialSize=50
jdbc.maxActive=100
jdbc.minIdle=10
jdbc.maxIdle=100
jdbc.maxWait=60000
jdbc.validationQuery=SELECT 1
jdbc.testOnBorrow=false
jdbc.testOnReturn=false
jdbc.testWhileIdle=true
jdbc.timeBetweenEvictionRunsMillis=30000
jdbc.minEvictableIdleTimeMillis=180000
jdbc.removeAbandoned=true
jdbc.removeAbandonedTimeout=1800
jdbc.logAbandoned=true
jdbc.poolPreparedStatements=false
jdbc.maxOpenPreparedStatements=20
jdbc.filters=stat

这个数据库里面有什么呢?就是我们的各个数据库的相关配置信息:

jdbc_config表生成的SQL语句: 

SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;

-- ----------------------------
-- Table structure for jdbc_config
-- ----------------------------
DROP TABLE IF EXISTS `jdbc_config`;
CREATE TABLE `jdbc_config`  (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `dbName` varchar(254) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `url` varchar(254) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `username` varchar(254) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `password` varchar(254) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `driverClassName` varchar(254) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
  `status` int(11) NULL DEFAULT NULL,
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 9520 CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Dynamic;

SET FOREIGN_KEY_CHECKS = 1;

PS:当然这个表结构不是必须固定这样的,等你看完整篇文章后就知道可以怎么修改了,表仅仅作为数据源信息的提供。

 

OK,数据源的表已经完成(当然你要把你的数据源信息都录入这个表里),接下来是修改xml文件,
 

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:beans="http://www.springframework.org/schema/beans"
       xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
         http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
         http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context-4.2.xsd
         http://www.springframework.org/schema/mvc
         http://www.springframework.org/schema/mvc/spring-mvc-4.2.xsd ">

    <!--配置和创建jdbc数据源 -->
    <context:property-placeholder location="classpath:jdbc.properties"
                                  ignore-unresolvable="true" />
    <bean id="Source" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
        <property name="driverClassName" value="${jdbc.driverClassName}" />
        <property name="url" value="${jdbc.url}" />
        <property name="username" value="${jdbc.username}" />
        <property name="password" value="${jdbc.password}" />
        <!-- 初始化连接大小 -->
        <property name="initialSize" value="${jdbc.initialSize}" />
        <!-- 连接池最大使用连接数量 -->
        <property name="maxActive" value="${jdbc.maxActive}" />
        <property name="maxIdle" value="${jdbc.maxIdle}"/>
        <!-- 连接池最小空闲 -->
        <property name="minIdle" value="${jdbc.minIdle}" />
        <!-- 获取连接最大等待时间 -->
        <property name="maxWait" value="${jdbc.maxWait}" />
        <property name="validationQuery" value="${jdbc.validationQuery}" />
        <property name="testOnBorrow" value="${jdbc.testOnBorrow}" />
        <property name="testOnReturn" value="${jdbc.testOnReturn}" />
        <property name="testWhileIdle" value="${jdbc.testWhileIdle}" />
        <!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
        <property name="timeBetweenEvictionRunsMillis" value="${jdbc.timeBetweenEvictionRunsMillis}" />
        <!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
        <property name="minEvictableIdleTimeMillis" value="${jdbc.minEvictableIdleTimeMillis}" />
        <!-- 打开removeAbandoned功能 -->
        <property name="removeAbandoned" value="${jdbc.removeAbandoned}" />
        <!-- 1800秒,也就是30分钟 -->
        <property name="removeAbandonedTimeout" value="${jdbc.removeAbandonedTimeout}" />
        <!-- 关闭abanded连接时输出错误日志 -->
        <property name="logAbandoned" value="${jdbc.logAbandoned}" />
        <!-- 监控数据库 -->
        <property name="filters" value="${jdbc.filters}" />

    </bean>
    <bean class="com.springmvc.dynamicsource.MutiDataSourceBean" id="dataSource">
        <property name="defaultTargetDataSource" ref="Source"/>
    </bean>
    <!-- 配置事务管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource" />
    </bean>
    <!-- 配置会话工厂SqlSessionFactory -->
    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <!-- 数据源 -->
        <property name="dataSource" ref="dataSource"/>
        <property name="configLocation" value="classpath:mybatis.xml" />
        <property name="mapperLocations" value="classpath:sqlmap/*Mapper.xml"/>
        <property name="typeAliasesPackage" value="com.springmvc.entity" />
    </bean>
    <!-- 在spring容器中配置mapper的扫描器产生的动态代理对象在spring的容器中自动注册,bean的id就是mapper类名(首字母小写)-->
    <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
        <!-- 指定扫描包的路径,就是mapper接口的路径,多个包中间以 半角逗号隔开   -->
        <property name="basePackage" value="com.springmvc.dao"/>
        <!-- 配置sqlSessionFactoryBeanName -->
        <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
    </bean>

</beans>

这里的配置信息,需要关注的是:
第一个蓝色框的 “Source”,是给数据库起的ID;
而第二个蓝框里面的“Source”,可以结合前面的defaultTargetDataSource,这里是配置项目跑起来默认连接的数据库;

红框的“dataSource”,非常重要,这是后面手动切换数据源获取信息的地方。 所以可以看到无论是事务管理器、sql工厂都是关联了这个dataSource。

 

OK,接下来是我们手动实现多数据源加载切换的java类:

后面提供手动切换数据库的方法类,DynamicDataSourceHolder.java:

public class DynamicDataSourceHolder {

	private static final ThreadLocal<String> contextHolder = new ThreadLocal<String>();

	/**
	 * @Description: 设置数据源类型 @param dataSourceType 数据库类型 @return void @throws
	 */
	public static void setDataSourceType(String dataSourceType) {
		contextHolder.set(dataSourceType);
	}

	/**
	 * @Description: 获取数据源类型 @param @return String @throws
	 */
	public static String getDataSourceType() {

		return contextHolder.get();
	}

	/**
	 * @Description: 清除数据源类型 @param @return void @throws
	 */
	public static void clearDataSourceType() {
		contextHolder.remove();
	}

}

用于jdbc连接数据库的类,JdbcPOJO.java:


import java.io.Serializable;

public class JdbcPOJO implements Serializable {

	private String driverClassName;

	private String id;
	private String url;
	private String username;
	private String password;

	public String getDriverClassName() {
		return driverClassName;
	}

	public void setDriverClassName(String driverClassName) {
		this.driverClassName = driverClassName;
	}

	public String getId() {
		return id;
	}

	public void setId(String id) {
		this.id = id;
	}

	public String getUrl() {
		return url;
	}

	public void setUrl(String url) {
		this.url = url;
	}

	public String getUsername() {
		return username;
	}

	public void setUsername(String username) {
		this.username = username;
	}

	public String getPassword() {
		return password;
	}

	public void setPassword(String password) {
		this.password = password;
	}

}

关键的多数据源加载类,MutiDataSourceBean.java:

import com.alibaba.druid.pool.DruidDataSource;
import com.springmvc.util.PropertyUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
import org.springframework.jdbc.datasource.lookup.DataSourceLookup;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;


public class MutiDataSourceBean extends AbstractRoutingDataSource implements ApplicationContextAware {

	private static Logger log = LoggerFactory.getLogger(MutiDataSourceBean.class);
	private static ApplicationContext ac;
	private static Map<String, String> buildDataSources = new HashMap();

	@Override
	public void afterPropertiesSet() {

		log.info("初始化多数据源");
		try {
			initailizeMutiDataSource();
		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
		log.info("多数据源加入spring容器中成功!");

		super.afterPropertiesSet();

	}

	@Override
	public void setApplicationContext(ApplicationContext ctx) throws BeansException {
		// TODO Auto-generated method stub
		ac = ctx;

	}

	private List<JdbcPOJO> getServers() {
		List<JdbcPOJO> list = new ArrayList();
		try {
			Class.forName(PropertyUtil.getPropery("jdbc.driverClassName"));
			Connection conn = DriverManager.getConnection(PropertyUtil.getPropery("jdbc.url").trim(),
					PropertyUtil.getPropery("jdbc.username").trim(), PropertyUtil.getPropery("jdbc.password").trim());

			PreparedStatement ps = conn.prepareStatement(
					"SELECT id,dbName,url,username,password,driverClassName FROM jdbc_config WHERE status=0");
			ResultSet rs = ps.executeQuery();

			while (rs.next()) {
				JdbcPOJO obj = new JdbcPOJO();
				obj.setId("Source" + rs.getString("dbName"));
				obj.setUrl(rs.getString("url"));
				obj.setUsername(rs.getString("username"));
				obj.setPassword(rs.getString("password"));
				obj.setDriverClassName(rs.getString("driverClassName"));

				list.add(obj);
			}

			if (rs != null) {
				rs.close();
			}

			if (ps != null) {
				ps.close();
			}

			if (conn != null) {
				conn.close();
			}

		} catch (Exception e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}

		return list;
	}

	public static boolean getSourcesExistsByKey(String key) {
		if (buildDataSources.containsKey(key)) {
			return true;
		} else {
			return false;
		}

	}

	private void initailizeMutiDataSource() throws Exception {

		List<JdbcPOJO> servers = getServers();
		
		DefaultListableBeanFactory acf = (DefaultListableBeanFactory) ac.getAutowireCapableBeanFactory();

		Map<Object, DruidDataSource> dsMap = new HashMap<Object, DruidDataSource>();

		for (JdbcPOJO obj : servers) {

			log.debug("初始化数据源:{}",obj.getId());
			System.out.println("数据源加载:   ----------"+obj.getId());
			DruidDataSource ds = new DruidDataSource();

			String id = obj.getId();
			ds.setDriverClassName(obj.getDriverClassName());
			ds.setUsername(obj.getUsername());
			ds.setUrl(obj.getUrl());
			ds.setPassword(obj.getPassword());

			ds.setInitialSize(Integer.parseInt(PropertyUtil.getPropery("jdbc.initialSize").trim()));
			ds.setMaxActive(Integer.parseInt(PropertyUtil.getPropery("jdbc.maxActive").trim()));
			ds.setMinIdle(Integer.valueOf(PropertyUtil.getPropery("jdbc.minIdle").trim()));
			ds.setMaxWait(Integer.valueOf(PropertyUtil.getPropery("jdbc.maxWait").trim()));
			ds.setTestOnBorrow(Boolean.parseBoolean(PropertyUtil.getPropery("jdbc.testOnBorrow").trim()));
			ds.setTestOnReturn(Boolean.parseBoolean(PropertyUtil.getPropery("jdbc.testOnReturn").trim()));
			ds.setTestWhileIdle(Boolean.parseBoolean(PropertyUtil.getPropery("jdbc.testWhileIdle").trim()));
			ds.setTimeBetweenEvictionRunsMillis(
					Long.valueOf(PropertyUtil.getPropery("jdbc.timeBetweenEvictionRunsMillis").trim()));
			ds.setMinEvictableIdleTimeMillis(
					Long.valueOf(PropertyUtil.getPropery("jdbc.minEvictableIdleTimeMillis").trim()));
			ds.setRemoveAbandoned(Boolean.parseBoolean(PropertyUtil.getPropery("jdbc.removeAbandoned").trim()));
			ds.setRemoveAbandonedTimeout(
					Integer.valueOf(PropertyUtil.getPropery("jdbc.removeAbandonedTimeout").trim()));
			ds.setLogAbandoned(Boolean.parseBoolean(PropertyUtil.getPropery("jdbc.logAbandoned").trim()));
			ds.setPoolPreparedStatements(
					Boolean.parseBoolean(PropertyUtil.getPropery("jdbc.poolPreparedStatements").trim()));
			ds.setMaxOpenPreparedStatements(
					Integer.parseInt(PropertyUtil.getPropery("jdbc.maxOpenPreparedStatements").trim()));
			ds.setFilters(PropertyUtil.getPropery("jdbc.filters").trim());
			acf.registerSingleton(id, ds);
			dsMap.put(id, ds);
			buildDataSources.put(id, id);
		}
		this.setTargetDataSources(dsMap);
		// setDefaultTargetDataSource(dsMap.get("Source"));//设置默认数据源
	}

	@Override
	protected Object determineCurrentLookupKey() {
		return DynamicDataSourceHolder.getDataSourceType();
	}

	@Override
	public void setDataSourceLookup(DataSourceLookup dataSourceLookup) {
		super.setDataSourceLookup(dataSourceLookup);
	}

	@Override
	public void setDefaultTargetDataSource(Object defaultTargetDataSource) {
		super.setDefaultTargetDataSource(defaultTargetDataSource);
	}

	@Override
	public void setTargetDataSources(Map targetDataSources) {
		super.setTargetDataSources(targetDataSources);
	}

}

简单介绍下上面这个类,项目跑起来后,简单的几步是:

  

 

然后读取数据库表jdbc_config里面的多数据源信息是:

顺便说下, 这里的id, 我项目里面写的是 Source拼接dbName的值(这个就是用于给我们切换数据源的id,规则可以自己修改):

OK,到这里多数据的整合其实已经完成了,我们来写个接口测试下,看看效果:

    @Autowired
    MessageboardService messageboardServiceImpl;
    @Autowired
    JdbcConfigService jdbcConfigServiceImpl;

    @RequestMapping(value = "/testDbSource", produces = "application/json; charset=utf-8")
    public void testDbSource() {

        System.out.println("当前数据源ID:"+DynamicDataSourceHolder.getDataSourceType());
        List<JdbcConfig> list1 = jdbcConfigServiceImpl.getList();
        System.out.println("获取结果数:"+list1.size()); //默认是 默认连接的数据库
        DynamicDataSourceHolder.setDataSourceType("Source"+"db1".trim()); //切换到数据源db1
        System.out.println("当前数据源ID:"+DynamicDataSourceHolder.getDataSourceType());
        Messageboard  message = messageboardServiceImpl.selectByPrimaryKey(3);
        System.out.println(message.toString());
        DynamicDataSourceHolder.setDataSourceType("Source"); //切换回来默认数据库
        System.out.println("当前数据源ID:"+DynamicDataSourceHolder.getDataSourceType());
        List<JdbcConfig> list2 = jdbcConfigServiceImpl.getList();
        System.out.println("获取结果数:"+list2.size()); //默认数据库随便查询一个表信息

    }

把项目跑起来:

用postman运行一下测试接口:

看下控制台输出情况:

刚开始,我们没有手动去切换数据源,可以看到从动态数据数据源控制器里获取的数据源为null,但是我们依然查出了数据,
这是因为我们项目里面配置了默认数据源,也就是这些数据是从默认数据库获取的。

接着我们切换到了数据源db1(规则是我们前面的数据源id规则:Source拼接dbName的值),可以看到从db1数据库成功查询出了相关的值:

最后,我们再切换回来默认数据库(默认数据源的ID就是xml文件里面的“Source”),随意切换后,查询也是正常的:

好了,到此,该篇操作多数据源的教程完毕。 

posted on 2022-11-08 07:35  小目标青年  阅读(74)  评论(0编辑  收藏  举报