baomidou的dynamic-datasource读写分离实现和加入AOP根据方法名选择库
文档
https://gitee.com/baomidou/dynamic-datasource-spring-boot-starter/wikis/pages
maven
- <dependency>
- <groupId>com.baomidou</groupId>
- <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
- <version>2.5.7</version>
- </dependency>
纯读写分离(mybatis环境)
场景:
- 在纯的读写分离环境,写操作全部是master,读操作全部是slave。
- 不想通过注解配置完成以上功能。
答:在mybatis环境下可以基于mybatis插件结合本数据源完成以上功能。 手动注入插件。
- @Bean
- public MasterSlaveAutoRoutingPlugin masterSlaveAutoRoutingPlugin(){
- return new MasterSlaveAutoRoutingPlugin();
- }
默认主库名称master,从库名称slave。
问题
我在配置好了之后,调试发现对数据库读的操作不得进入MasterSlaveAutoRoutingPlugin,而且进入了默认的库。只有写进入了MasterSlaveAutoRoutingPlugin中。当然也可以默认为从库,但是感觉就不是很好。
于是我自定义了一个aop切面来,来完成库的选择,代码如下:
- import java.lang.reflect.Method;
- import org.aspectj.lang.JoinPoint;
- import org.aspectj.lang.annotation.After;
- import org.aspectj.lang.annotation.Aspect;
- import org.aspectj.lang.annotation.Before;
- import org.aspectj.lang.annotation.Pointcut;
- import org.aspectj.lang.reflect.MethodSignature;
- import org.springframework.context.annotation.Lazy;
- import org.springframework.core.annotation.Order;
- import org.springframework.stereotype.Component;
- import com.baomidou.dynamic.datasource.annotation.DS;
- import com.baomidou.dynamic.datasource.toolkit.DynamicDataSourceContextHolder;
- import lombok.extern.java.Log;
- /**
- * Copyright: Copyright (c) 2019
- * <p> 说明:动态数据源配置 </P>
- *
- * @version: V1.0
- * @author: BianPeng
- *
- */
- @Aspect
- @Component
- @Order(0)
- @Lazy(false)
- @Log
- public class DataSourceAop{
- private static final String MASTER = "master";
- private static final String SLAVE = "slave";
- @Pointcut("execution(* com.buybit.power.service..*.*(..)) || execution(* com.baomidou.mybatisplus.extension.service..*.*(..))")
- public void checkArgs() {
- }
- // 这里切到你的方法目录
- @Before("checkArgs()")
- public void process(JoinPoint joinPoint) throws NoSuchMethodException, SecurityException {
- String methodName = joinPoint.getSignature().getName();
- if (methodName.startsWith("get")
- || methodName.startsWith("count")
- || methodName.startsWith("find")
- || methodName.startsWith("list")
- || methodName.startsWith("select")
- || methodName.startsWith("check")
- || methodName.startsWith("page")) {
- log.info("当前执行的库:"+SLAVE);
- DynamicDataSourceContextHolder.push(SLAVE);
- } else {
- log.info("当前执行的库:"+MASTER);
- DynamicDataSourceContextHolder.push(MASTER);
- }
- }
- @After("checkArgs()")
- public void afterAdvice(){
- DynamicDataSourceContextHolder.clear();
- }
- }
但是发现,baomidou/dynamic-datasource自带的@DS没失去了着用,于是我把有@DS的类和方法排除掉,代码入下:
- import java.lang.reflect.Method;
- import org.aspectj.lang.JoinPoint;
- import org.aspectj.lang.annotation.After;
- import org.aspectj.lang.annotation.Aspect;
- import org.aspectj.lang.annotation.Before;
- import org.aspectj.lang.annotation.Pointcut;
- import org.aspectj.lang.reflect.MethodSignature;
- import org.springframework.context.annotation.Lazy;
- import org.springframework.core.annotation.Order;
- import org.springframework.stereotype.Component;
- import com.baomidou.dynamic.datasource.annotation.DS;
- import com.baomidou.dynamic.datasource.toolkit.DynamicDataSourceContextHolder;
- import lombok.extern.java.Log;
- /**
- * Copyright: Copyright (c) 2019
- * <p> 说明:动态数据源配置 </P>
- *
- * @version: V1.0
- * @author: BianPeng
- *
- */
- @Aspect
- @Component
- @Order(0)
- @Lazy(false)
- @Log
- public class DataSourceAop{
- private static final String MASTER = "master";
- private static final String SLAVE = "slave";
- @Pointcut("execution(* com.buybit.power.service..*.*(..)) || execution(* com.baomidou.mybatisplus.extension.service..*.*(..))")
- public void checkArgs() {
- }
- // 这里切到你的方法目录
- @Before("checkArgs()")
- public void process(JoinPoint joinPoint) throws NoSuchMethodException, SecurityException {
- String methodName = joinPoint.getSignature().getName();
- Class clazz = joinPoint.getTarget().getClass();
- if(clazz.isAnnotationPresent(DS.class)){
- //获取类上注解
- return;
- }
- String targetName = clazz.getSimpleName();
- Class[] parameterTypes =
- ((MethodSignature)joinPoint.getSignature()).getMethod().getParameterTypes();
- Method methdo = clazz.getMethod(methodName,parameterTypes);
- if (methdo.isAnnotationPresent(DS.class)) {
- return;
- }
- if (methodName.startsWith("get")
- || methodName.startsWith("count")
- || methodName.startsWith("find")
- || methodName.startsWith("list")
- || methodName.startsWith("select")
- || methodName.startsWith("check")
- || methodName.startsWith("page")) {
- log.info("当前执行的库:"+SLAVE);
- DynamicDataSourceContextHolder.push(SLAVE);
- } else {
- log.info("当前执行的库:"+MASTER);
- DynamicDataSourceContextHolder.push(MASTER);
- }
- }
- @After("checkArgs()")
- public void afterAdvice(){
- DynamicDataSourceContextHolder.clear();
- }
- }
这样可以让你有@DS的注解依然生效,而且也会根据方法名来自动切换数据源。
baomidou的dynamic-datasource读写分离实现和加入AOP根据方法名选择库的更多相关文章
- SSM/SSH框架的MySQL 读写分离实现的一种简单方法
简介 MySQL已经是使用最为广泛的一种数据库,往往实际使用过程中,为实现高可用及高性能,项目会采用主丛复制的方式实现读写分离.MySQL本身支持复制,通过简单的配置即可实现一主多从的配置,具体实现可 ...
- 【EFCore框架】支持多数库 、支持读写分离框架、支持事务提交保存、跨库实现事务
1.支持多数库: 数据库上下文实现 多数据支持: 2. 支持读写分离框架(支持多从库随机) 工作单元(以及仓储)实现读写分离 配置文件: 3. 支持事务提交保存 工作单元 实现事务数据统一提交保存 具 ...
- laravel(lumen)配置读写分离后,强制读主(写)库数据库,解决主从延迟问题
在Model里面加上下面这句,强制读主(写)库数据库,解决主从延迟问题. public static function boot() { //清空从连接,会自动使用主连接 DB::connection ...
- Springboot + Mysql8实现读写分离
在实际的生产环境中,为了确保数据库的稳定性,我们一般会给数据库配置双机热备机制,这样在master数据库崩溃后,slave数据库可以立即切换成主数据库,通过主从复制的方式将数据从主库同步至从库,在业务 ...
- 从零开始学 Java - Spring AOP 实现主从读写分离
深刻讨论为什么要读写分离? 为了服务器承载更多的用户?提升了网站的响应速度?分摊数据库服务器的压力?就是为了双机热备又不想浪费备份服务器?上面这些回答,我认为都不是错误的,但也都不是完全正确的.「读写 ...
- JAVA-mysql读写分离插件介绍
kingshard是一个由Go开发高性能MySQL Proxy项目,kingshard在满足基本的读写分离的功能上,致力于简化MySQL分库分表操作:能够让DBA通过kingshard轻松平滑地实现M ...
- Spring Boot2(四):使用Spring Boot多数据源实现读写分离
前言 实际业务场景中,不可能只有一个库,所以就有了分库分表,多数据源的出现.实现了读写分离,主库负责增改删,从库负责查询.这篇文章将实现Spring Boot如何实现多数据源,动态数据源切换,读写分离 ...
- Mybatis多数据源读写分离(注解实现)
#### Mybatis多数据源读写分离(注解实现) ------ 首先需要建立两个库进行测试,我这里使用的是master_test和slave_test两个库,两张库都有一张同样的表(偷懒,喜喜), ...
- 读写分离很难吗?springboot结合aop简单就实现了
目录 前言 环境部署 开始项目 注意 參考: 前言 入职新公司到现在也有一个月了,完成了手头的工作,前几天终于有时间研究下公司旧项目的代码.在研究代码的过程中,发现项目里用到了Spring Aop来实 ...
- 原理解密 → Spring AOP 实现动态数据源(读写分离),底层原理是什么
开心一刻 女孩睡醒玩手机,收到男孩发来一条信息:我要去跟我喜欢的人表白了! 女孩的心猛的一痛,回了条信息:去吧,祝你好运! 男孩回了句:但是我没有勇气说不来,怕被打! 女孩:没事的,我相信你!此时女孩 ...
随机推荐
- 19 Transformer 解码器的两个为什么(为什么做掩码、为什么用编码器-解码器注意力)
博客配套视频链接: https://space.bilibili.com/383551518?spm_id_from=333.1007.0.0 b 站直接看 配套 github 链接:https:// ...
- 007 Python、Anaconda、pip、Pycharm、Jupyter都是啥?
博客配套视频链接: https://space.bilibili.com/383551518?spm_id_from=333.1007.0.0 b 站直接看 配套 github 链接:https:// ...
- k8s的pod的理解
pod共享相同的IP地址和端口空间. 这意味着在同一 pod中的容器运行的 多个进程需要注意不能绑定到相同的端口号, 否则会导致端口冲突, 但这只涉及同一pod中的容器. 由于每个pod都有独立的端口 ...
- 从2s优化到0.1s,我用了这5步
前言 分类树查询功能,在各个业务系统中可以说随处可见,特别是在电商系统中. 但就是这样一个简单的分类树查询功能,我们却优化了5次. 到底是怎么回事呢? 背景 我们的网站使用了SpringBoot推荐的 ...
- 在 KubeSphere 中开启新一代云原生数仓 Databend
作者:尚卓燃(https://github.com/PsiACE),Databend 研发工程师,Apache OpenDAL (Incubating) PPMC. 前言 Databend 是一款完全 ...
- 快速部署kafka集群和可视化界面
3台机器部署kafka集群的分布 10.1.161.111 -> kafka(1个节点)+ zookeeper(一个节点)+ kowl(一个节点) 10.1.161.112 -> kafk ...
- Java中重写equals并重写hashcode方法的描述
1.两个对象的hashCode()相同,equals()不一定为true (1)重写equals()的同时,重写hashCode() a.定义Person类,私有属性name,age:有参构造,set ...
- 同步完善Docker常用操作命令
镜像 images_name 表示镜像名 con_name表示容器名 #获取镜像 docker pull images_name #查看已有的docker镜像 docker images #删除镜像 ...
- 【斩虫】Hadoop中作业执行刚开始就挂掉的两种情况
开门见山. 最近在搭建基于 Hadoop 3.3.6 的高可用集群时,遇到了虽然守护进程能正常启动,但是提交 WordCount 示例程序后作业没有办法启动执行的情况(刚开始就挂了),查看日志发现主要 ...
- SpringBoot学习 _02 _配置单数据源(1)
SpringBoot学习 _02 _配置单数据源(1) 1.通过IDEA创建SpringBoot项目 直接到选择依赖这一步,选择如图所示的 五个依赖即可 2.配置文件的编写 此处说明一下,这个数据库连 ...