SpringBoot如何在运行时动态添加数据源
作者:听说我很强 时间:2023-11-13 21:36:40
此方案适用于解决springboot项目运行时动态添加数据源,非静态切换多数据源!!!
一、多数据源应用场景:
1.配置文件配置多数据源,如默认数据源:master,数据源1:salve1...,运行时动态切换已配置的数据源(master、salve1互相切换),无法在运行时动态添加配置文件中未配置的数据源。
2.配置一个默认数据源,运行时动态添加新数据源使用(本博客适用于此场景)
二、解决方案:
Spring提供了AbstractRoutingDataSource用于动态路由数据源,第一种场景继承AbstractRoutingDataSource类并覆写其protected abstract Object determineCurrentLookupKey()即可;
而第二种场景我们直接覆写protected DataSource determineTargetDataSource方法即可。原理可看下AbstractRoutingDataSource对应源码,比较简单,不做赘述。
直接上干货:
import com.fizz.utils.spring.SpringUtils;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;
import javax.sql.DataSource;
public class DynamicDataSource extends AbstractRoutingDataSource {
private static final ThreadLocal<DataSource> dataSource = ThreadLocal.withInitial(() -> (DataSource) SpringUtils.getBean("defaultDataSource"));
public static void setDataSource(DataSource dataSource) {
DynamicDataSource.dataSource.set(dataSource);
}
public static DataSource getDataSource() {
return DynamicDataSource.dataSource.get();
}
@Override
protected Object determineCurrentLookupKey() {
return null;
}
@Override
protected DataSource determineTargetDataSource() {
return getDataSource();
}
public static void clear() {
DynamicDataSource.dataSource.remove();
}
}
import com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceBuilder;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import javax.sql.DataSource;
import java.util.HashMap;
@Configuration
public class DataSourceConfig {
@Bean
@ConfigurationProperties("spring.datasource.druid")
public DataSource defaultDataSource() {
return DruidDataSourceBuilder.create().build();
}
@Bean
@Primary
public DynamicDataSource dataSource() {
DynamicDataSource dynamicDataSource = new DynamicDataSource();
dynamicDataSource.setTargetDataSources(new HashMap<>());
return dynamicDataSource;
}
}
使用时直接调用DynamicDataSource.setDataSource(DataSource dataSource)方法即可,使用完后调用DynamicDataSource.clear()防止内存泄漏并重置默认数据源。
附上详细使用方法:
DruidDataSource druidDataSource = new DruidDataSource();
druidDataSource.setUrl("jdbc:mysql://localhost:3306/sys?characterEncoding=utf-8&useSSL=false&serverTimezone=UTC&useAffectedRows=true");
druidDataSource.setUsername("root");
druidDataSource.setPassword("root");
DynamicDataSource.setDataSource(druidDataSource);
此时数据源已切换到druidDataSource ,调用自己的业务方法即可。
使用完后调用DynamicDataSource.clear();重置为默认数据源。
附上工具类SpringUtils :
import lombok.Getter;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
@Component
public final class SpringUtils implements ApplicationContextAware {
@Getter
private static ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
if (SpringUtils.applicationContext == null) {
SpringUtils.applicationContext = applicationContext;
}
}
public static <T> T getBean(Class<T> clazz) {
return SpringUtils.applicationContext.getBean(clazz);
}
public static Object getBean(String name) {
return SpringUtils.applicationContext.getBean(name);
}
public static String getProperty(String key) {
return SpringUtils.applicationContext.getEnvironment().getProperty(key);
}
}
来源:https://www.cnblogs.com/tswhq/p/11668078.html
标签:spring,boot,运行,动态,添加,数据源
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
c# 代理模式
2022-02-19 09:48:31
![](https://img.aspxhome.com/file/2023/3/107773_0s.gif)
微信随机生成红包金额算法java版
2023-07-27 16:01:39
java实现统计字符串中字符及子字符串个数的方法示例
2022-10-14 13:47:40
![](https://img.aspxhome.com/file/2023/2/129102_0s.png)
冒泡排序算法原理及JAVA实现代码
2022-08-13 10:30:40
Android编程之文件读写操作与技巧总结【经典收藏】
2023-10-16 02:20:52
浅谈关于spring profile的误解
2021-07-25 05:48:43
Java中线程的等待与唤醒_动力节点Java学院整理
2023-07-17 04:03:05
![](https://img.aspxhome.com/file/2023/4/57694_0s.png)
android 自定义圆角button效果的实例代码(自定义view Demo)
2022-07-13 11:35:09
![](https://img.aspxhome.com/file/2023/1/139471_0s.png)
一文搞懂String的intern()方法
2022-05-28 12:05:35
![](https://img.aspxhome.com/file/2023/8/125608_0s.webp)
Java+Swing实现医院管理系统的完整代码
2023-03-17 00:40:21
![](https://img.aspxhome.com/file/2023/1/74991_0s.jpg)
Java新特性之Nashorn_动力节点Java学院整理
2022-07-31 17:18:13
![](https://img.aspxhome.com/file/2023/2/62822_0s.png)
Java使用新浪微博API通过账号密码方式登陆微博的实例
2023-09-23 05:35:38
快速学习C# 设计模式之职责链模式
2022-03-28 17:47:45
![](https://img.aspxhome.com/file/2023/5/67565_0s.jpg)
Android开发手册TextView属性实现效果盘点
2022-03-10 09:29:53
![](https://img.aspxhome.com/file/2023/2/109192_0s.jpg)
c#和avascript加解密之间的互转代码分享
2022-03-22 10:15:28
C#数组初始化简析
2022-01-02 11:55:12
UnityUI中绘制线状统计图
2022-12-03 14:30:43
![](https://img.aspxhome.com/file/2023/6/82146_0s.jpg)
C#调用SQLite的方法实例分析
2022-09-25 06:02:22
c# 判断是否为空然后赋值的4种实现方法
2021-06-06 22:01:12
![](https://img.aspxhome.com/file/2023/9/104239_0s.jpg)
C#中的尾递归与Continuation详解
2021-12-05 16:35:15