开心一刻
2023年元旦,我妈又开始了对我的念叨
妈:你到底想多少岁结婚
我:60
妈:60,你想找个多大的
我:找个55的啊,她55我60,结婚都有退休金,不用上班不用生孩子,不用买车买房,成天就是玩儿
我:而且一结婚就是白头偕老,多好
我妈直接一大嘴巴子呼我脸上
data:image/s3,"s3://crabby-images/e01ea/e01ea9facd65365f41ecb347864d3bb619ac3503" alt=""
需求背景
最近接到一个需求,需要从两个数据源获取数据,然后进行汇总展示
一个数据源是 MySQL ,另一个数据源是 SQL Server
楼主是一点都不慌的,因为我写过好几篇关于数据源的文章
spring集成mybatis实现mysql读写分离
原理解密 → Spring AOP 实现动态数据源(读写分离),底层原理是什么
Spring 下,关于动态数据源的事务问题的探讨
我会慌?
data:image/s3,"s3://crabby-images/0c4e3/0c4e35cc7a5c97a056b66a37d2a163f3da327d14" alt=""
但还是有点小拒绝,为什么了?
自己实现的话,要写的东西还是很多,要写 AOP ,还要实现 AbstractRoutingDataSource ,还要用到 ThreadLocal ,...
如果考虑更远一些,事务、数据源之间的嵌套等等,要如何保证正确?
但好在这次需求只是查询,然后汇总,问题就简单很多了,但还是觉得有点小繁琐
data:image/s3,"s3://crabby-images/5dab5/5dab593183cbc05fa0648a2c30f4948285d01a57" alt=""
当然,如上只是楼主的臆想
有小伙伴可能会问道:能不能合到一个数据源?
楼主只能说:别问了,再问就不礼貌了
data:image/s3,"s3://crabby-images/54fb3/54fb3a3c9ae49129ddc5e46ca80c4bbbdcdccab4" alt=""
既然改变不了,那就盘它
data:image/s3,"s3://crabby-images/22418/224187c96d58287d5c2269337752a74dc670247f" alt=""
难道就没有现成的多数据源工具?
因为用到了 Mybatis-Plus ,楼主试着 Google 了一下
data:image/s3,"s3://crabby-images/1a7e6/1a7e691a4f2275d196fd97ef1f64df3a55fdcce8" alt=""
直接一发入魂,眼前一黑,不对,是眼前一亮!
感觉就是它了!
MyBatis-Plus 多数据源
关于苞米豆(baomidou),我们最熟悉的肯定是 MyBatis-Plus
但旗下还有很多其他优秀的组件
data:image/s3,"s3://crabby-images/9b6a7/9b6a73fd783cbda6d789869a25045a3c3b1eea1e" alt=""
多数据源就是其中一个,今天我们就来会会它
数据源准备
用 docker 准备一个 MySQL 和 SQL Server ,图省事,两个数据库服务器放到同个 docker 下了
有小伙伴会觉得放一起不合适,有单点问题!
楼主只是为了演示,纠结那么细,当心敲你狗头
data:image/s3,"s3://crabby-images/99458/994586a7c9c1658260d4d1dbbe10e1aa6c55d00e" alt=""
MySQL 版本: 8.0.27
data:image/s3,"s3://crabby-images/c77ae/c77ae099c83885e391f8de1e797ac901da01b021" alt=""
建库: datasource_mysql ,建表: tbl_user ,并插入初始化数据
data:image/s3,"s3://crabby-images/e0f05/e0f05a005ca74af689d31917b7f728b20fba3814" alt="" data:image/s3,"s3://crabby-images/8bdce/8bdce277043852b5cc34353ff5b6afee93e19db3" alt="" - CREATE DATABASE datasource_mysql;
- USE datasource_mysql;
- CREATE TABLE tbl_user (
- id INT UNSIGNED NOT NULL AUTO_INCREMENT,
- user_name VARCHAR(50),
- PRIMARY KEY(id)
- );
- INSERT INTO tbl_user(user_name) VALUES('张三'),('李四');
复制代码 View Codedata:image/s3,"s3://crabby-images/bc161/bc1618993f50e465a69c5b86dc4069dd65814d12" alt=""
SQL Server 版本: Microsoft SQL Server 2017 ... ,是真长,跟楼主一样长!
data:image/s3,"s3://crabby-images/2f007/2f007fccb912dc74d633c0881b0bdd5820f3805a" alt=""
建库: datasource_mssql ,建表: tbl_order ,并插入初始化数据
data:image/s3,"s3://crabby-images/e0f05/e0f05a005ca74af689d31917b7f728b20fba3814" alt="" data:image/s3,"s3://crabby-images/8bdce/8bdce277043852b5cc34353ff5b6afee93e19db3" alt="" - CREATE DATABASE datasource_mssql;
- USE datasource_mssql;
- CREATE TABLE tbl_order(
- id BIGINT PRIMARY KEY IDENTITY(1,1),
- order_no NVARCHAR(50),
- created_at DATETIME NOT NULL DEFAULT(GETDATE()),
- updated_at DATETIME NOT NULL DEFAULT(GETDATE())
- );
- INSERT INTO tbl_order(order_no) VALUES('123456'),('654321');
复制代码 View Codedata:image/s3,"s3://crabby-images/3ca2b/3ca2bf09d3317a4dacd18191f0d7a1f15a9e1b41" alt=""
dynamic-datasource 使用
基于 spring-boot 2.2.10.RELEASE 、 mybatis-plus 3.1.1 搭建
dynamic-datasource-spring-boot-starter 也是 3.1.1
依赖很简单, pom.xml
data:image/s3,"s3://crabby-images/e0f05/e0f05a005ca74af689d31917b7f728b20fba3814" alt="" data:image/s3,"s3://crabby-images/8bdce/8bdce277043852b5cc34353ff5b6afee93e19db3" alt="" - <?xml version="1.0" encoding="UTF-8"?>
- <project xmlns="http://maven.apache.org/POM/4.0.0"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
- <modelVersion>4.0.0</modelVersion>
- <groupId>com.lee</groupId>
- <artifactId>mybatis-plus-dynamic-datasource</artifactId>
- <version>1.0-SNAPSHOT</version>
- <parent>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-starter-parent</artifactId>
- <version>2.2.10.RELEASE</version>
- </parent>
- <properties>
- <java.version>1.8</java.version>
- <maven.compiler.source>8</maven.compiler.source>
- <maven.compiler.target>8</maven.compiler.target>
- <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
- <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
- <mybatis-plus-boot-starter.version>3.1.1</mybatis-plus-boot-starter.version>
- <mssql-jdbc.version>6.2.1.jre8</mssql-jdbc.version>
- </properties>
- <dependencies>
- <dependency>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-starter-web</artifactId>
- </dependency>
- <dependency>
- <groupId>org.projectlombok</groupId>
- <artifactId>lombok</artifactId>
- </dependency>
- <dependency>
- <groupId>com.baomidou</groupId>
- <artifactId>mybatis-plus-boot-starter</artifactId>
- <version>${mybatis-plus-boot-starter.version}</version>
- </dependency>
- <dependency>
- <groupId>com.baomidou</groupId>
- <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
- <version>${mybatis-plus-boot-starter.version}</version>
- </dependency>
-
- <dependency>
- <groupId>mysql</groupId>
- <artifactId>mysql-connector-java</artifactId>
- </dependency>
-
- <dependency>
- <groupId>com.microsoft.sqlserver</groupId>
- <artifactId>mssql-jdbc</artifactId>
- <version>${mssql-jdbc.version}</version>
- </dependency>
- </dependencies>
- </project>
复制代码 View Code 配置也很简单, application.yml
data:image/s3,"s3://crabby-images/e0f05/e0f05a005ca74af689d31917b7f728b20fba3814" alt="" data:image/s3,"s3://crabby-images/8bdce/8bdce277043852b5cc34353ff5b6afee93e19db3" alt="" - server:
- port: 8081
- spring:
- application:
- name: dynamic-datasource
- datasource:
- dynamic:
- datasource:
- mssql_db:
- driver-class-name: com.microsoft.sqlserver.jdbc.SQLServerDriver
- url: jdbc:sqlserver://10.5.108.225:1433;DatabaseName=datasource_mssql;IntegratedSecurity=false;ApplicationIntent=ReadOnly;MultiSubnetFailover=True
- username: sa
- password: Root#123456
- mysql_db:
- driver-class-name: com.mysql.cj.jdbc.Driver
- url: jdbc:mysql://10.5.108.225:3306/datasource_mysql?useSSL=false&useUnicode=true&characterEncoding=utf-8&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=Asia/Shanghai
- username: root
- password: 123456
- primary: mssql_db
- strict: false
- mybatis-plus:
- mapper-locations: classpath:mappers/*.xml
- configuration:
- log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
复制代码 View Code 然后在对应的类或者方法上加上注解 DS("数据源名称") 即可,例如
data:image/s3,"s3://crabby-images/831c9/831c92fd296a837bc8a1a05e45a916702d632ee2" alt=""
我们来看下效果
data:image/s3,"s3://crabby-images/b8391/b8391e975dbb838403911bcfe9e72913deb46a81" alt=""
是不是很神奇?
data:image/s3,"s3://crabby-images/8f2d1/8f2d15627de3c55fe71b24816ad412f3f57ae23d" alt=""
完整代码:mybatis-plus-dynamic-datasource
原理探究
@DS 用于指定数据源,可以注解在方法上或类上,同时存在则采用就近原则 方法上注解 优先于 类上注解
这可不是我瞎说,官方文档就是这么写的
data:image/s3,"s3://crabby-images/dbdf0/dbdf0eb5e66455e848d30166cb70d76767d721cf" alt=""
难道一个 @DS 就有如此强大的功能?你们不信,我也不信,它背后肯定有人!
data:image/s3,"s3://crabby-images/42d10/42d1009aabf9c3b84a1498628bfbad86299a3df6" alt=""
那么我们就来揪一揪背后的它
怎么揪了,这又是个难题,我们先打个断点,看一下调用栈
data:image/s3,"s3://crabby-images/ff60b/ff60bc3ac5ef1b80a92d937965eb5ea42cdbc45e" alt=""
点一下,瞬间高潮了,不是,是瞬间清醒了
data:image/s3,"s3://crabby-images/96149/961491943ba9ae6c71c447538a0db7e11898d1a6" alt=""
红线框住的,分 2 点:1: determineDatasource ,2: DynamicDataSourceContextHolder.push
我们先看 determineDatasource
data:image/s3,"s3://crabby-images/b5a50/b5a50b1e717b12eb9d839b3b24cca97c3ede17c3" alt=""
1、获取 Method 对象
2、该方法上是否有 DS 注解,有则取方法的 DS 注解,没有则取方法对应的类上的 DS 注解;这个看明白了没?
3、获取注解的值,也就是 @DS("mysql_db") 中的 mysql_db
4、如果数据源名不为空并且数据原名以动态前缀(#)开头,则你们自己去跟 dsProcessor.determineDatasource
data:image/s3,"s3://crabby-images/b0794/b0794007955de78c77be5b0ef2f816b27c76a644" alt=""
否则则直接返回数据源名
针对案例的话,这里肯定是返回类上的数据源名(方法上没有指定数据源,也没有以动态前缀开头)
我们再来看看 DynamicDataSourceContextHolder.push
data:image/s3,"s3://crabby-images/dfbfb/dfbfb3ac06b7d12205bb751b874d0ce6d48f3a9e" alt=""
很简单,但 LOOKUP_KEY_HOLDER 很有意思
data:image/s3,"s3://crabby-images/fe13e/fe13e90a51caf63188868c75ed1102d70e652b57" alt=""
是一个栈,而非楼主在spring集成mybatis实现mysql读写分离 采用的
data:image/s3,"s3://crabby-images/76b53/76b538a85c9262ac69502539ab7cfd2c3c2f7875" alt=""
至于为什么,人家注释已经写的很清楚了,试问楼主的实现能满足一级一级数据源切换的调用场景吗?
但不管怎么说, LOOKUP_KEY_HOLDER 的类型还是 ThreadLocal
接下来该分析什么?
data:image/s3,"s3://crabby-images/a7c2f/a7c2fd321e6ccf00f4f6ce206ee814df45a8c013" alt=""
我们回顾下:原理解密 → Spring AOP 实现动态数据源(读写分离),底层原理是什么
直接跳到总结
data:image/s3,"s3://crabby-images/17a4b/17a4b9bdf41b43be056da399d4bc723e009aaf56" alt=""
框住的 3 条,上面的 2 条在上面已经分析过了把,是不是?你回答是就完事了
data:image/s3,"s3://crabby-images/8ffd1/8ffd1ccf6ee43e8a7cc95677edae70d48111038b" alt=""
注意,楼主的 DynamicDataSource 是自实现的类,继承了 spring-jdbc 的 AbstractRoutingDataSource
data:image/s3,"s3://crabby-images/37401/37401552a04696ae6a032f3360e9d5ea036adce6" alt=""
那我们就找 AbstractRoutingDataSource 的实现类呗
data:image/s3,"s3://crabby-images/48cc0/48cc01554177f9d60a4e402c3ddb6274b950efdf" alt=""
发现它就一个实现类,并且是在 spring-jdbc 下,而不是在 com.baomidou 下
莫非苞米豆有自己的 AbstractRoutingDataSource ? 我们来看看 AbstractDataSource 的实现类有哪些
data:image/s3,"s3://crabby-images/ffe16/ffe16ed7794605365ad7b3128a6244bef9ab545d" alt=""
看到了没,那么我们接下来就分析它
data:image/s3,"s3://crabby-images/5e21f/5e21fa462baecce48c508f1909aecf5a03690073" alt=""
内容很简单,最重要的 determineDataSource 还是个抽象方法,那没办法了,看它有哪些子类实现
data:image/s3,"s3://crabby-images/cd603/cd6039793ba8830cbfbf6a2bff348197ebc01533" alt=""
DynamicRoutingDataSource 的 determineDataSource 方法如下
data:image/s3,"s3://crabby-images/48765/487654dea04e062ac4114d3e4b330cd7099655de" alt=""
DynamicDataSourceContextHolder 有没有感觉到熟悉?
想想它的 ThreadLocal LOOKUP_KEY_HOLDER ,回忆上来了没?
出栈,获取到当前的数据源名;接下来该分析谁了?
那肯定是 getDataSource 方法
data:image/s3,"s3://crabby-images/0dc9f/0dc9f7b9b7faeec38d5e9058c37bf17104a6aa32" alt=""
1、如果数据源为空,那么直接返回默认数据源,对应配置文件中的
data:image/s3,"s3://crabby-images/93f2c/93f2c4e6ef90e0cc1385466c7ff97ac57316f4d5" alt=""
2、分组数据源,我们的示例代码那么简单,应该没涉及到这个,先不管
3、所有数据源,是一个 LinkHashMap ,key 是 数据源名 ,value 是数据源
可想而知,我们示例的数据源获取就是从该 map 获取的
4、是否启用严格模式,默认不启动。严格模式下未匹配到数据源直接报错,,非严格模式下则使用默认数据源 primary 所设置的数据源
5、对应 4,未开启严格模式,未匹配到数据源则使用 primary 所设置的数据源
那现在又该分析谁?肯定是 dataSourceMap 的值是怎么 put 进去的
我们看哪些地方用到了 dataSourceMap
data:image/s3,"s3://crabby-images/1edf9/1edf900595ef36b0de987973ebe2e32f29bffacd" alt=""
发现就一个地方进行了 put
data:image/s3,"s3://crabby-images/6730e/6730e15dfcf5f099ffe980b7dcdee4cdf8588d90" alt=""
那这个 addDataSource 方法又在哪被调用了?
data:image/s3,"s3://crabby-images/d22dc/d22dc312d9d57410056e0b1c4d86163258c364e9" alt=""
DynamicRoutingDataSource 实现了 InitializingBean ,所以在启动过程中,它的 afterPropertiesSet 方法会被调用,至于为什么,大家自行去查阅
接下来该分析什么?那肯定是 Map dataSources = provider.loadDataSources();
我们跟进 loadDataSources() ,发现有两个类都有该方法
data:image/s3,"s3://crabby-images/117da/117da97ecce30a82a895370b9cb68e188b0bf341" alt=""
那么我们应该跟谁?有两种方法
1、凭感觉,我们的配置文件是 yml
2、打断点,重新启动项目,一目了然
data:image/s3,"s3://crabby-images/913d5/913d5831e1f949fa9f7ac57f59dcaa0b4e35a63f" alt=""
YmlDynamicDataSourceProvider 的 loadDataSources 方法如下
data:image/s3,"s3://crabby-images/00df2/00df2537b6f8d1f0108626e9f04f84b439a10a0d" alt=""
(这里留个疑问: dataSourcePropertiesMap 存放的是什么,值是如何 put 进去的?)
继续往下跟 createDataSourceMap 方法
data:image/s3,"s3://crabby-images/9d060/9d060dc8a61dd3c2f724f08087af76c92314362f" alt=""
1、配置文件中的数据源属性,断点下就很清楚了
data:image/s3,"s3://crabby-images/04f67/04f679b22a86e58606c82c607b3838bae2b14375" alt=""
2、根据数据源属性创建数据源,然后放进 dataSourceMap 中
创建数据源的过程就不跟了,感兴趣的自行去研究
至此,不知道大家清楚了没? 我反正是晕了
data:image/s3,"s3://crabby-images/33a7a/33a7a5dd044c6aa399a028c5e31cb3a94be2e1e7" alt=""
总结
1、万变不离其宗,多数据源的原理是不变的
原理解密 → Spring AOP 实现动态数据源(读写分离),底层原理是什么
2、苞米豆的多数据源的自动配置类
com.baomidou.dynamic.datasource.spring.boot.autoconfigure.DynamicDataSourceAutoConfiguration
这个配置类很重要,很多重要的对象都是在这里注入到 Spring 容器中的
关于自动配置,大家可参考:springboot2.0.3源码篇 - 自动配置的实现,发现也不是那么复杂
3、遇到问题,不要立马一头扎进去,自己实现,多查查,看是否有现成的第三方实现
自己实现,很容易踩别人踩过的坑,容易浪费时间;另外局限性太大,不易拓展,毕竟一人之力有限
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作! |