SpringBoot动态切换数据源
SpringBoot整合多数据源,动态添加新数据源并切换
- 1.需求
- 2.创建数据源配置类
- 3.切换数据源
- 4.切换数据源管理类
- 5.使用案例
- 5.AOP切面拦截
1.需求
低代码服务需要给多套系统进行功能配置,要求表结构必须生成在对应系统的数据库中,所以表结构的生成需要动态的获取目标系统的数据库信息,切换当前数据源到目标数据库,然后再生成。
2.创建数据源配置类
package com.tyq.datasource.config.datasource;import com.alibaba.druid.pool.DruidDataSource;
import com.tyq.datasource.config.Properties;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;import javax.annotation.Resource;
import javax.sql.DataSource;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;/*** 向Spring容器中注入DruidConfiguration** @author 谭永强*/
@Configuration
@EnableAutoConfiguration(exclude = {DataSourceAutoConfiguration.class})
public class DruidConfiguration {@Resourceprivate Properties properties;/*** 数据源(默认)*/public DataSource dataSourceToDefault() {DruidDataSource datasource = new DruidDataSource();try {datasource.setQueryTimeout(0);datasource.setUrl("jdbc:oracle:thin:@192.168.0.30:1521/test2");datasource.setUsername("SCPS");datasource.setPassword("yldtscps");} catch (SQLException e) {e.printStackTrace();}return datasource;}@Bean@Primarypublic DataSource dataSource() {Map<Object, Object> dataSourceMap = new HashMap<>(2);dataSourceMap.put("default", dataSourceToDefault());DynamicDatasource dynamicDatasource = new DynamicDatasource();// 注入目标数据源,如果有多个数据源,直接加入map即可dynamicDatasource.setTargetDataSources(dataSourceMap);// 注入默认数据源dynamicDatasource.setDefaultTargetDataSource(dataSourceToDefault());return dynamicDatasource;}
}
3.切换数据源
SpringBoot动态切换数据源主要依靠AbstractRoutingDataSource类,这个抽象类中有一个属性为targetDataSources。该属性为Map结构,所有需要切换的数据源都存放在其中,根据指定的KEY进行切换。
在AbstractRoutingDataSource源码中,获取数据库连接是通过this.determineTargetDataSource().getConnection()去获取的,而this.determineTargetDataSource()方法中的DataSource则是通过determineCurrentLookupKey()方法返回的key值去前面的dynamicDatasource.setTargetDataSources(dataSourceMap);设置进去的map中查找的。
所以我们需要重写determineCurrentLookupKey()方法,如下:
package com.tyq.datasource.config.datasource;import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;/*** 动态数据源** @author 谭永强* @date 2023-08-03*/
public class DynamicDatasource extends AbstractRoutingDataSource {@Overrideprotected Object determineCurrentLookupKey() {return DynamicDatasourceHolder.getDataSource();}
}
4.切换数据源管理类
数据源属于公共资源,考虑到多线程的情况下,我们将数据源存储在【ThreadLocal】中,保证线程隔离。
package com.tyq.datasource.config.datasource;import java.util.Queue;
import java.util.concurrent.ArrayBlockingQueue;/*** 数据源切换管理** @author 谭永强* @date 2023-08-03*/
public class DynamicDatasourceHolder {/*** 保存数据源的映射*/private final static ThreadLocal<String> CONTEXT_HOLDER = new ThreadLocal<>();public static String getDataSource() {return CONTEXT_HOLDER.get();}public static void setDataSource(String dataSourceKey) {CONTEXT_HOLDER.set(dataSourceKey);}public static void removeDataSource() {CONTEXT_HOLDER.remove();}
}
5.使用案例
在Controller中通过DynamicDatasourceHolder指定当前需要使用哪个数据源,具体使用如下:
@RequestMapping("findById")
public User findById(String userId) {//指定数据源DynamicDatasourceHolder.setDataSource("default");if (ObjectUtils.isEmpty(userId)) {throw new ParamValidateException("userId不能为空");}User user = userService.findById(userId);//移除当前数据源DynamicDatasourceHolder.removeDataSource();return user;
}
上述案例中已经实现了如果动态切换数据源的过程,但是在实际开发中还是太繁琐,比如每个接口都必须添加相关切换数据源的代码,对代码的侵入性太高,下面我们通过AOP的形式去优化切换数据源的过程。
5.AOP切面拦截
通过AOP切面对方法的前置和后置做切换数据源的操作,这样就降低了与业务代码耦合、
<!--AOP-->
<dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.9.6</version>
</dependency>
注意:如果项目中无法导入@Aspect,则需要添加上述依赖。
package com.tyq.datasource.aop;import com.alibaba.druid.pool.DruidDataSource;
import com.tyq.datasource.config.Properties;
import com.tyq.datasource.config.datasource.DynamicDatasource;
import com.tyq.datasource.config.datasource.DynamicDatasourceHolder;
import com.tyq.datasource.config.exception.ParamValidateException;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.stereotype.Component;
import org.springframework.util.ObjectUtils;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.sql.DataSource;
import java.sql.SQLException;
import java.util.HashMap;
import java.util.Map;/*** 动态数据源切换AOP** @author 谭永强* @date 2023-08-03*/
@Aspect
@Component
public class DynamicDatasourceAop {@Autowiredprotected ApplicationContext applicationContext;@Resourceprivate Properties properties;/*** 定义切点*/@Pointcut("execution (* com.tyq.datasource.controller.*.*(..))")public void pointcut() {}/*** 前置处理*/@Before(value = "pointcut()")public void beforeAdvice() {// 接收到请求,记录请求内容ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();if (ObjectUtils.isEmpty(attributes)) {return;}HttpServletRequest request = attributes.getRequest();String dataSourceKey = request.getParameter("dataSourceKey");if (ObjectUtils.isEmpty(dataSourceKey)) {throw new ParamValidateException("dataSourceKey不能为空!");}//获取当前动态数据源DynamicDatasource dynamicDatasource = applicationContext.getBean(DynamicDatasource.class);//所有已连接的数据源集合Map<Object, DataSource> resolvedDataSources = dynamicDatasource.getResolvedDataSources();if (ObjectUtils.isEmpty(resolvedDataSources.get(dataSourceKey))) {//此处为模拟到数据库中查询数据源信息的过程,查询到数据源信息后,创建数据源并添加到动态数据源管理中//动态加入新的数据源DataSource dataSource = getDataSource();Map<Object, Object> dataSourceMap = new HashMap<>(resolvedDataSources.size() + 1);dataSourceMap.putAll(resolvedDataSources);dataSourceMap.put("three", dataSource);dynamicDatasource.setTargetDataSources(dataSourceMap);}//动态切换数据源DynamicDatasourceHolder.setDataSource(dataSourceKey);}/*** 后置处理*/@AfterReturning(value = "pointcut()")public void afterAdvice() {// 接收到请求,记录请求内容ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();if (ObjectUtils.isEmpty(attributes)) {return;}HttpServletRequest request = attributes.getRequest();String dataSourceKey = request.getParameter("dataSourceKey");if (ObjectUtils.isEmpty(dataSourceKey)) {throw new ParamValidateException("dataSourceKey不能为空!");}System.out.println("后置处理:" + dataSourceKey);//删除当前数据源DynamicDatasourceHolder.removeDataSource();}/*** 模拟成功查询到数据源并创建DataSource** @return 数据源*/public DataSource getDataSource() {DruidDataSource datasource = new DruidDataSource();try {datasource.setQueryTimeout(0);datasource.setUrl("jdbc:oracle:thin:@192.168.0.59:1526/qstest");datasource.setUsername("SCPS");datasource.setPassword("yldtscps");} catch (SQLException e) {e.printStackTrace();}return datasource;}
}相关文章:
SpringBoot动态切换数据源
SpringBoot整合多数据源,动态添加新数据源并切换 1.需求2.创建数据源配置类3.切换数据源4.切换数据源管理类5.使用案例5.AOP切面拦截 1.需求 低代码服务需要给多套系统进行功能配置,要求表结构必须生成在对应系统的数据库中,所以表结构的生成…...
[C++项目] Boost文档 站内搜索引擎(4): 搜索的相关接口的实现、线程安全的单例index接口、cppjieba分词库的使用、综合调试...
有关Boost文档搜索引擎的项目的前三篇文章, 已经分别介绍分析了: 项目背景: 🫦[C项目] Boost文档 站内搜索引擎(1): 项目背景介绍、相关技术栈、相关概念介绍…文档解析、处理模块parser的实现: 🫦[C项目] Boost文档 站内搜索引擎(2): 文档文本解析模块…...
SAP ABAP元素域值描述通过函数(DD_DOMVALUE_TEXT_GET)获取
代码如下: PERFORM FRM_GET_DOMVALUE_TEXT USING ZMMD_ZFLZQ <GFS_DATA>-ZFLZQ CHANGING <GFS_DATA>-ZZQTEXT .IF <GFS_DATA>-ZXYLX IS NOT INITIAL .PERFORM FRM_GET_DOMVALUE_TEXT USING ZMMD_ZXYLX <GFS_DATA>-ZXYLX CHANGING <GFS_…...
原型模式与享元模式:提升系统性能的利器
原型模式和享元模式,前者是在创建多个实例时,对创建过程的性能进行调优;后者是用减 少创建实例的方式,来调优系统性能。这么看,你会不会觉得两个模式有点相互矛盾呢? 在有些场景下,我们需要重复…...
uniapp封装手写签名
组件代码 cat-signature <template><view v-if"visibleSync" class"cat-signature" :class"{visible:show}" touchmove.stop.prevent"moveHandle"><view class"mask" tap"close" /><view c…...
掌握 JVM 调优命令
常用命令 1、jps查看当前 java 进程2、jinfo实时查看和调整 JVM 配置参数3、jstat查看虚拟机统计信息4、jstack查看线程堆栈信息5、jmap查看堆内存的快照信息 JVM 日常调优总结起来就是:首先通过 jps 命令查看当前进程,然后根据 pid 通过 jinfo 命令查看…...
扩增子分析流程——Lotus2: 一行命令完成所有分析
为什么介绍lotus2 因为快,作者比较了lotus2流程和qiime2、dada2、vsearch等,lotus2的速度最快、占用内存最小。 因为方便,只需要一行代码,即可完成全部分析。 lotus2 -i Example/ -m Example/miSeqMap.sm.txt -o myTestRun而且分…...
微服务 云原生:搭建 Harbor 私有镜像仓库
Harbor官网 写在文前: 本文中用到机器均为虚拟机 CentOS-7-x86_64-Minimal-2009 镜像。 基础设施要求 虚拟机配置达到最低要求即可,本次系统中使用 docker 24.0.4、docker-compose 1.29.2。docker 及 docker-compose 的安装可以参考上篇文章 微服务 &am…...
Ceph入门到精通-远程开发Windows下使用SSH密钥实现免密登陆Linux服务器
工具: win10、WinSCP 服务器生成ssh密钥: 打开终端,使账号密码登录,输入命令 ssh-keygen -t rsa Winscp下载 Downloading WinSCP-6.1.1-Setup.exe :: WinSCP window 生成密钥 打开powershell ssh-keygen -t rsa 注意路径 …...
APP外包开发的开发语言对比
在开发iOS APP时有两种语言可以选择,Swift(Swift Programming Language)和 Objective-C(Objective-C Programming Language),它们是两种不同的编程语言,都被用于iOS和macOS等苹果平台的软件开发…...
基于Python++PyQt5马尔科夫模型的智能AI即兴作曲—深度学习算法应用(含全部工程源码+测试数据)
目录 前言总体设计系统整体结构图系统流程图 运行环境Python 环境PC环境配置 模块实现1. 钢琴伴奏制作1)和弦的实现2)和弦级数转为当前调式音阶3)根据预置节奏生成伴奏 2. 乐句生成1)添加音符2)旋律生成3)节…...
Android中简单封装Livedata工具类
Android中简单封装Livedata工具类 前言: 之前讲解过livedata和viewmodel的简单使用,也封装过room工具类,本文是对livedata的简单封装和使用,先是封装了一个简单的工具类,然后实现了一个倒计时工具类的封装. 1.LiveD…...
国内大模型在局部能力上已超ChatGPT
中文大模型正在后来居上,也必须后来居上。 数科星球原创 作者丨苑晶 编辑丨大兔 从GPT3.5彻底出圈后,大模型的影响力开始蜚声国际。一段时间内,国内科技公司可谓被ChatGPT按在地上打,毫无还手之力。 彼时,很多企业…...
监控设置ip地址怎么设置
监控设备的IP地址设置是保障监控系统正常工作的基础。通过设置IP地址,我们可以确定监控设备在局域网内的位置,并远程访问监控设备进行实时查看、存储视频等操作。下面虎观代理小二二将介绍具体步骤。 方法一: 和电脑连接在一起,…...
力扣:56. 合并区间(Python3)
题目: 以数组 intervals 表示若干个区间的集合,其中单个区间为 intervals[i] [starti, endi] 。请你合并所有重叠的区间,并返回 一个不重叠的区间数组,该数组需恰好覆盖输入中的所有区间 。 来源:力扣(Lee…...
最小二乘问题和非线性优化
最小二乘问题和非线性优化 0.引言1.最小二乘问题2.迭代下降法3.最速下降法4.牛顿法5.阻尼法6.高斯牛顿(GN)法7.莱文贝格马夸特(LM)法8.鲁棒核函数 0.引言 转载自此处,修正了一点小错误。 1.最小二乘问题 在求解 SLAM 中的最优状态估计问题时,我们一般…...
Selenium/webdriver原理解析
最近在看一些底层的东西。driver翻译过来是驱动,司机的意思。如果将webdriver比做成司机,竟然非常恰当。 我们可以把WebDriver驱动浏览器类比成出租车司机开出租车。在开出租车时有三个角色: 乘客:他/她告诉出租车司机去哪里&…...
多用户跨境B2B2C商城后台管理系统快速搭建
搭建一个多用户跨境B2B2C商城后台管理系统需要考虑多个方面,包括系统架构设计、用户权限管理、商品管理、订单管理、支付管理、物流管理等。搭建步骤如下: 1. 系统架构设计 首先,需要设计一个稳定可靠的系统架构。选择一个适合B2B2C商城的商…...
MySQL 优化
问题描述 MySQL 的性能优化分为四个部分: 硬件和操作系统层面的优化架构设计层面的优化MySQL 程序配置优SQL 优化 一、硬件及操作系统层面优化 从硬件层面来说,影响 Mysql 性能的因素有,CPU、可用内存大小、磁盘读写速度、 网络带宽。 从操作…...
VMware Workstation及CentOS-7虚机安装
创建新的虚机: 选择安装软件(这里选的是桌面版,也可以根据实际情况进行选择) 等待检查软件依赖关系 选择安装位置,自主配置分区 创建一个普通用户 安装完成后重启 点击完成配置,进入登陆界面…...
十分钟搞定登录原型:用快马AI快速生成全站登录应用前端与后端
今天想和大家分享一个快速搭建全站登录应用原型的经验。最近在做一个新项目,需要验证登录模块的流程设计,传统开发方式至少要花一两天时间配置前后端环境,但这次尝试用InsCode(快马)平台的AI生成功能,十分钟就搞定了可交互的原型。…...
Air8101 搭载 RGB 直驱与 AirUI 适配工业电容屏开发
Air8101专为工业电容屏优化设计,RGB接口可直驱各类尺寸LCD电容屏,最高可支持1024*720分辨率屏,无需额外转接,大幅降低硬件开发成本。 一、硬件直驱: 目前正在支持完善:480*272分辨率:4.3寸屏800…...
收藏必备!小白程序员轻松入门大模型,解锁医学AI新技能
收藏必备!小白程序员轻松入门大模型,解锁医学AI新技能 大语言模型在医疗健康领域应用初见成效,但存在知识体系固化的局限。本文介绍了检索增强生成(RAG)技术,该技术模拟医生查阅最新文献的工作逻辑…...
提升嵌入式开发效率:用快马平台一键生成串口通信等常用模块代码
作为一名嵌入式开发者,我经常需要和串口通信打交道。无论是调试信息输出、设备间通信还是固件升级,UART都是最常用的外设之一。但每次新项目都要重新写一遍串口初始化、中断处理这些重复性代码,实在有点浪费时间。最近发现InsCode(快马)平台能…...
别再为ChatTTS声音飘忽发愁了!手把手教你用Python代码+高质量.pt音色文件,生成稳定语音
用Python和优质音色文件打造稳定语音合成体验 语音合成技术正在改变内容创作的方式,但很多开发者在实际使用ChatTTS时都会遇到一个共同的困扰——生成的语音音色飘忽不定,每次输出都像开盲盒。这种不稳定性严重影响了专业场景下的使用体验,比…...
ai赋能c语言开发:让快马平台自动生成文件io与链表管理代码
AI赋能C语言开发:让快马平台自动生成文件IO与链表管理代码 最近在做一个C语言的通讯录管理系统项目,需要实现联系人信息的增删改查功能,并且要求数据能够持久化保存。作为一个有经验的开发者,我决定尝试用InsCode(快马)平台的AI辅…...
新手福音:在快马平台用openclaw启动项目迈出机器人开发第一步
作为一名刚接触机器人开发的新手,第一次听说openclaw启动项目时,我完全不知道从何入手。机械爪控制、PWM信号、硬件通信这些术语听起来就让人头大。好在发现了InsCode(快马)平台,它帮我用最直观的方式理解了整个流程。 项目框架搭建 平台提供…...
HS2-HF Patch汉化补丁:3分钟实现Honey Select 2游戏完全汉化
HS2-HF Patch汉化补丁:3分钟实现Honey Select 2游戏完全汉化 【免费下载链接】HS2-HF_Patch Automatically translate, uncensor and update HoneySelect2! 项目地址: https://gitcode.com/gh_mirrors/hs/HS2-HF_Patch 如果你正在寻找一款能够彻底解决Honey …...
Mem Reduct多语言界面配置指南:从基础设置到高级应用
Mem Reduct多语言界面配置指南:从基础设置到高级应用 【免费下载链接】memreduct Lightweight real-time memory management application to monitor and clean system memory on your computer. 项目地址: https://gitcode.com/gh_mirrors/me/memreduct 功能…...
UABEA:Unity游戏资源编辑与分析的终极解决方案
UABEA:Unity游戏资源编辑与分析的终极解决方案 【免费下载链接】UABEA c# uabe for newer versions of unity 项目地址: https://gitcode.com/gh_mirrors/ua/UABEA 在Unity游戏开发和模组制作领域,处理Asset Bundle资源文件是每个开发者都会面临的…...
