springboot,druid动态数据源切换

关键字:springboot,druid数据库连接池,两个数据源(可以切换成多个),事务管理

关于druid简介传送门:https://github.com/alibaba/druid/wiki/%E5%B8%B8%E8%A7%81%E9%97%AE%E9%A2%98
在这里插入图片描述

具体分为四步:

1,ThreadLocal维护分支信息,

/*** @ClassName DataSourceContextHolder* @Description: 设置threadlocal数据元信息* @Author gjq* @Date 2024/2/29* @Version V1.0**/
public class DataSourceContextHolder {/**使用Threadlocal维护变量,threadlocal为每个使用该变量的现成提供独立的变量副本,所以每一个线程可以独立的改变自己的副本,而不会影响其他线程使用对应的副本**/private static final ThreadLocal<String> HOLDER = new InheritableThreadLocal<>();public static void setDataSource(String key) {HOLDER.set(key);}public static String getDataSource() {return HOLDER.get();}public static void clearDataSource() {HOLDER.remove();}}

2,创建数据源设置方法,初始化数据源信息

package com.pg.ga.datasource;import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;import javax.sql.DataSource;
import java.util.Map;/*** @ClassName DynamicDataSource* @Description:* @Author gjq* @Date 2024/2/29* @Version V1.0**/
public class DynamicDataSource extends AbstractRoutingDataSource {public DynamicDataSource(DataSource dataSource, Map<Object, Object> targetDataSources) {super.setDefaultTargetDataSource(dataSource);super.setTargetDataSources(targetDataSources);super.afterPropertiesSet();}@Overrideprotected Object determineCurrentLookupKey() {return DataSourceContextHolder.getDataSource();}}

3,多数据源配置,及枚举值定义

枚举值定义:
package com.pg.ga.datasource;public enum DataSourceType {MASTER,SLAVE
}
package com.pg.ga.conf;import com.alibaba.druid.pool.DruidDataSource;
import com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceBuilder;
import com.baomidou.mybatisplus.extension.spring.MybatisSqlSessionFactoryBean;
import com.pg.ga.datasource.DataSourceType;
import com.pg.ga.datasource.DynamicDataSource;
import org.apache.ibatis.session.SqlSessionFactory;
import org.mybatis.spring.annotation.MapperScan;
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 org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.core.io.support.ResourcePatternResolver;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.PlatformTransactionManager;import javax.sql.DataSource;
import java.io.IOException;
import java.util.*;/*** @ClassName PgConfig* @Description:类描述* @Author gjq* @Date 2024/2/29* @Version V1.0**/
@Configuration
@MapperScan({"com.pg.ga.dao"})
public class PgConfig {@Bean@ConfigurationProperties("spring.datasource.druid.master")public DruidDataSource masterDataSource() {return DruidDataSourceBuilder.create().build();}@Bean@ConfigurationProperties("spring.datasource.druid.slave")// @ConditionalOnProperty(prefix = "spring.datasource.druid.slave", name = "enable", havingValue = "true")public DataSource slaveDataSource() {return DruidDataSourceBuilder.create().build();}@Bean(name = "dynamicDataSource")@Primarypublic DynamicDataSource dataSource(DataSource masterDataSource, DataSource slaveDataSource) {Map<Object, Object> targetDataSource = new HashMap<>();targetDataSource.put(DataSourceType.MASTER.name(), masterDataSource);targetDataSource.put(DataSourceType.SLAVE.name(), slaveDataSource);return new DynamicDataSource(masterDataSource(), targetDataSource);}@Beanpublic SqlSessionFactory sqlSessionFactory(DynamicDataSource dynamicDataSource) throws Exception {//核心就是这个,替换原始的SqlSessionFactoryBean 用mybatisSqlSessionFactoryBean即可MybatisSqlSessionFactoryBean factoryBean = new MybatisSqlSessionFactoryBean();factoryBean.setDataSource(dynamicDataSource);//设置mapper.xml位置路径factoryBean.setMapperLocations(resolveMapperLocations());return factoryBean.getObject();}public Resource[] resolveMapperLocations() {ResourcePatternResolver resourcePatternResolver = new PathMatchingResourcePatternResolver();List<String> mapperLocations = new ArrayList<>();mapperLocations.add("classpath*:mapper/*.xml");List<Resource> resourceList = new ArrayList<>();if (null != mapperLocations) {for (String mapperLocation : mapperLocations) {try {Resource[] mappers = resourcePatternResolver.getResources(mapperLocation);resourceList.addAll(Arrays.asList(mappers));} catch (IOException e) {e.printStackTrace();}}}return resourceList.toArray(new Resource[resourceList.size()]);}/*** @MethodName:* @Description: 配置@Transactional注解事务* @Param:* @Return:* @Author: gjq* @Date: 2024/2/29**/@Beanpublic PlatformTransactionManager transactionManager(DynamicDataSource dynamicDataSource) {return new DataSourceTransactionManager(dynamicDataSource);}}

4,动态切换注解定义

package com.pg.ga.datasource;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface DataSource {/**@MethodName:@Description: 切换数据源名称@Param:@Return:@Author: gjq@Date: 2024/2/29**/DataSourceType value() default DataSourceType.MASTER;
}
切面实现
package com.pg.ga.datasource;import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;import java.lang.reflect.Method;/*** @ClassName DataSourceAspect* @Description:类描述* @Author gjq* @Date 2024/2/29* @Version V1.0**/
@Aspect
@Component
@Order(1)
public class DataSourceAspect {//先声明切点@Pointcut("@annotation(com.pg.ga.datasource.DataSource)")public void dsPointCut() {System.out.println(11111);}@Around("dsPointCut()")public Object around(ProceedingJoinPoint point) throws Throwable{MethodSignature signature = (MethodSignature) point.getSignature();Method method = signature.getMethod();DataSource dataSource = method.getAnnotation(DataSource.class);if(null!=dataSource){DataSourceContextHolder.setDataSource(dataSource.value().name());}try{return point.proceed();}finally {DataSourceContextHolder.clearDataSource();}}
}

数据库配置信息.yaml文件配置

server :port : 8090##项目名字配置#servlet :#  context-path : /demotomcat :uri-encoding : UTF-8#xx 报错修改的地方max-connections: 200000max-http-form-post-size: 9000000threads:max: 128min-spare: 5
spring :# 环境 dev|test|prodprofiles :active : dev#引入其他配置文件,例如ftpHX 未配置文件application-ftpHX.yml#include: ftpHX,ftpCloudservlet:multipart:#设置总上传的数据大小max-request-size: 100MB#单个文件大小maxFileSize : 30MB#xx 报错修改的地方max-connections: 200000max-http-post-size: 9000000#热部署模块devtools:restart:#热部署开关enabled: true#指定热部署的目录additional-paths: src/main/java#指定目录不更新exclude: test/**mvc:   #静态文件static-path-pattern : /static/**pathmatch:matching-strategy: ant_path_matcher#模板引擎thymeleaf:model: HTML5prefix: classpath:/templates/suffix: .html#指定编码encoding: utf-8#禁用缓存 默认falsecache: falsejackson:time-zone: GMT+8date-format: yyyy-MM-dd HH:mm:ssredis:ssl: falsedatabase: 0host: 127.0.0.1port: 6379password: timeout: 1000lettuce:pool:max-active: 200max-wait: -1max-idle: 10min-idle: 0# 控制台输出sql、下划线转驼峰
mybatis-plus:mapper-locations: classpath:/mapper/*.xmltypeAliasesPackage: com.pg.ga.model# 控制台输出sql、下划线转驼峰configuration:log-impl: org.apache.ibatis.logging.stdout.StdOutImplmap-underscore-to-camel-case: true#pagehelper分页插件
pagehelper:helperDialect: mysqlreasonable: truesupportMethodsArguments: trueparams: count=countSql
application-dev.yaml配置信息
#dev环境  mysql7.0
spring:datasource:type: com.alibaba.druid.pool.DruidDataSourcedriverClassName: org.postgresql.Driver#druid连接池配置druid:#主库数据源master:url: jdbc:postgresql://X.X.X.X:30811/ticketusername: postgrespassword: postgres#备数据源 #关闭slave:enabled: falseurl: jdbc:postgresql://X.X.X.X:30811/ticketusername: postgrespassword: postgres#配置初始化连接数大小initial-size: 10# 最大连接数maxActive: 50#最小连接数minIdle: 10#获取连接等待超时时间maxWait: 5000poolPreparedStatements: true #是否缓存preparedStatement,也就是PSCache。PSCache对支持游标的数据库性能提升巨大,比如说oracle。在mysql下建议关闭。maxPoolPreparedStatementPerConnection-size: 20validationQuery: SELECT 1 FROM DUALvalidationQueryTimeout: 20000testOnBorrow: false #申请连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能。testOnReturn: false #归还连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能。testWhileIdle: true #建议配置为true,不影响性能,并且保证安全性。申请连接的时候检测,如果空闲时间大于timeBetweenEvictionRunsMillis,执行validationQuery检测连接是否有效。timeBetweenEvictionRunsMillis: 60000 #配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒minEvictableIdleTimeMillis: 300000  #一个连接在池中最小生存的时间,单位是毫秒maxEvictableIdleTimeMillis: 900000  # 配置一个连接在池中最大生存的时间,单位是毫秒#StatViewServlet配置。(因为暴露的监控信息比较敏感,支持密码加密和访问ip限定)webStatFilter:enabled: truestatViewServlet:enabled: trueurlPattern: /druid/*#可以增加访问账号密码【去掉注释就可以】#login-username: admin#login-password: adminfilter:stat:enabled: true# 慢SQL记录log-slow-sql: trueslow-sql-millis: 1000merge-sql: truewall:config:multi-statement-allow: true
测试controller
package com.pg.ga.controller;import com.alibaba.fastjson.JSONObject;
import com.pg.ga.model.TicketBall;
import com.pg.ga.service.TickService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
@RequestMapping("/crud")
public class Crud {@AutowiredTickService tickService;@GetMapping("/master")public String master() {TicketBall ticketBall = tickService.selectMaster();System.out.println(JSONObject.toJSON(ticketBall));return null;}@GetMapping("/slave")public String create() {TicketBall ticketBall = tickService.selectSlave();System.out.println(JSONObject.toJSON(ticketBall));return null;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/269427.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

利用文件实现进程间共享数据

概述 文件可以存储任何非结构化字节序列&#xff0c;这个比较简单&#xff0c;就一个写一个读&#xff1b;学习到此&#xff0c;留个记录&#xff0c;以后可以直接抄代码&#xff0c;哈哈 Demo代码 #include <fstream> #include <iostream> #include <thread&…

代码随想录day10(2)字符串:反转字符串Ⅱ (leetcode541)

题目要求&#xff1a;给定一个字符串 s 和一个整数 k&#xff0c;从字符串开头算起, 每计数至 2k 个字符&#xff0c;就反转这 2k 个字符中的前 k 个字符。如果剩余字符少于 k 个&#xff0c;则将剩余字符全部反转。如果剩余字符小于 2k 但大于或等于 k 个&#xff0c;则反转前…

python实现ElGamal算法

ElGamal公钥密码算法是在密码协议中有着重要应用的一类公钥密码算法&#xff0c;基于公钥密码体制和椭圆曲线加密体系&#xff0c;其安全性是基于有限域上离散对数学问题的难解性。至今仍是一个安全性良好的公钥密码算法。既可用于加密又可用于数字签名的公钥密码体制。 数字签…

【vue.js】文档解读【day 1】 | 模板语法2

如果阅读有疑问的话&#xff0c;欢迎评论或私信&#xff01;&#xff01; 本人会很热心的阐述自己的想法&#xff01;谢谢&#xff01;&#xff01;&#xff01; 文章目录 模板语法JavaScript表达式仅支持表达式调用函数&#xff1f;受限的全局访问 指令参数动态参数动态参数中…

Vanna-ai -基于RAG的TextToSql实现方案

官方连接&#xff1a;Vanna.AI - Personalized AI SQL Agent 1.背景 基于大模型的TextToSql的关键为给大模型提供正确有效的数据库信息及问题&#xff0c;以提升大模型生成sql的正确率。database_info question形成prompt&#xff0c;但是实际中通常会遇到一个问题&#xff…

C# WinForm AndtUI第三方库 Tree控件使用记录

环境搭建 1.在NuGet中搜索AndtUI并下载至C# .NetFramework WinForm项目。 2.添加Tree控件至窗体。 使用方法集合 1.添加节点、子节点 using AntdUI; private void UpdateTreeView() {Tree tvwTestnew Tree();TreeItem rootTreeItem;TreeItem subTreeItem;Dictionary<str…

openGL 透视投影矩阵

openGL 透视投影矩阵 什么是透视投影&#xff1f; 模型都是3D的&#xff0c;但屏幕是2D的。如何将3D空间投影到2D平面&#xff0c;还能保持深度的视觉效果&#xff1f;在OpenGL中&#xff0c;采用透视投影矩阵作用顶点来实现&#xff0c;即完成缩放、选择、位移之后&#xff…

CSS_实现三角形和聊天气泡框

如何用css画出一个三角形 1、第一步 写一个正常的盒子模型&#xff0c;先给个正方形的div&#xff0c;便于观察&#xff0c;给div设置宽高和背景颜色 <body><div class"box"></div> </body> <style>.box {width: 100px;height: 100px…

Windows已经安装了QT 6.3.0,如何再安装一个QT 5.12

要在Windows上安装Qt 5.12&#xff0c;您可以按照以下步骤操作&#xff1a; 下载Qt 5.12&#xff1a;访问Qt官方网站或其他可信赖的来源&#xff0c;下载Qt 5.12的安装包。 下载安装地址 下载安装详细教程 安装问题点 qt安装时“Error during installation process(qt.tools…

opencart3 添加速卖通商品脚本

非爬虫&#xff0c;只能把速卖通商品信息拿下来解析插入到自己的项目里。 刚接触opencart3没多久&#xff0c;有一些新项目需要添加商品&#xff0c;每次手动从速卖通复制信息又很慢&#xff0c;就自己写了一个脚本。 思路&#xff1a;速卖通商品详情页有一段数据包含了几乎所…

Springboot 过滤器、拦截器、全局异常处理

Springboot 过滤器、拦截器、全局异常处理 一 过滤器&#xff08;Filter&#xff09; 过滤器是JavaWeb三大组件&#xff08;Servlet&#xff0c;Filter&#xff0c;Listener&#xff09;之一。 Filter可以把对资源的请求拦截下来&#xff0c;从而实现一些功能。 注意&#…

这本书太好了!150页就能让你上手大模型应用开发

如果问个问题&#xff1a;有哪些产品曾经创造了伟大的奇迹&#xff1f;ChatGPT 应该会当之无愧入选。仅仅发布 5 天&#xff0c;ChatGPT 就吸引了 100 万用户——当然&#xff0c;数据不是关键&#xff0c;关键是其背后的技术开启了新的 AI 狂潮&#xff0c;成为技术变革的点火…

Ubantu 18.04 配置固定IP

1.首先在终端里输入命令,将你的网关和ip&#xff0c;记下来 ifconfig 2. 执行命令&#xff1a; sudo gedit /etc/network/interfaces 3.在弹出来的框里输入 auto后面的就是网关&#xff0c;address是你虚拟机的ip&#xff0c;gateway是你的网关ip&#xff0c;netmask是你的子…

非阻塞实现高效键盘扫描功能(STM32F4XX)

目录 概述 1 原理分析 1.1 技术背景 1.2 系统硬件 1.3 STM32 IO&#xff08;输入模式&#xff09;寄存器分析 1.3.1 输入IO的功能描述 1.3.2 输入配置 1.3.3 GPIO 寄存器&#xff08;输入模式相关&#xff09; 1.3.3.1 GPIO 端口模式寄存器 1.3.3.2 GPIO 端口上拉/下拉…

挑战给女神节送礼物,怎么寄快递才能快速的送到他手中呢?

马上就是三八女神节了&#xff0c;怎么样&#xff1f;你给心爱的她或者敬爱的她准备礼物了吗&#xff0c;如果已经准备好&#xff0c;你该怎么送给她呢&#xff1f;是当面送给她&#xff1f;还是通过快递打包送给她呢&#xff1f;这里推荐使用闪侠惠递寄快递发货给他吧&#xf…

Zookeeper3:客户端命令

文章目录 客户端命令连接服务端Zookeeper客户端内置命令 ls - 节点信息 客户端命令 连接服务端Zookeeper //客户端连接服务端zookeeper 默认连的本机2181端口的zookeeper cd /opt/module/zookeeper-3.9.1/bin && sh zkCli.sh//客户端连接远程服务端zookeeper cd /op…

MySQL高可用性攻略:快速搭建MySQL主从复制集群 !

MySQL高可用性攻略&#xff1a;快速搭建MySQL主从复制集群 &#xff01; MySQL基础知识&#xff1a;介绍MySQL数据库的基本概念和常用命令&#xff0c;如何创建数据库、表、用户和权限管理等。 MySQL安装教程&#xff1a;Centos7 安装MySQL5.7.29详细安装手册 MySQL数据类型&…

【Docker】Docker:解析容器化技术的利器与在Linux中的关键作用

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a;Linux ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 前言 正文 Docker 是什么&#xff1f; Docker 的作用 Docker 在 Linux 中的重要性 结语 我的其他博客 前言 随着软件开发的不断发展&…

猴子吃桃问题(python版)

文章预览&#xff1a; 题目python解法一&#xff1a;运行结果 python解法二&#xff1a;运行结果 python解法三&#xff1a;运行结果 题目 猴子吃桃问题&#xff1a;猴子第一天摘下若干个桃子&#xff0c;当即吃了一半&#xff0c;还不过瘾&#xff0c;又多吃了一个。 第二天早…

计算机网络-网络安全(二)

1.应用层安全协议&#xff1a; S-HTTP或SHTTP&#xff08;Sec HTTP&#xff09;&#xff0c;安全超文本传输协议&#xff0c;是HTTP扩展&#xff0c;使用TCP的80端口。HTTPS&#xff1a;HTTPSSL&#xff0c;使用TCP的443端口。和TLS&#xff08;传输层安全标准&#xff09;是双…