文档

https://gitee.com/baomidou/dynamic-datasource-spring-boot-starter/wikis/pages

maven

  1.  
    <dependency>
  2.  
    <groupId>com.baomidou</groupId>
  3.  
    <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
  4.  
    <version>2.5.7</version>
  5.  
    </dependency>

纯读写分离(mybatis环境)

场景:

  1. 在纯的读写分离环境,写操作全部是master,读操作全部是slave。
  2. 不想通过注解配置完成以上功能。

答:在mybatis环境下可以基于mybatis插件结合本数据源完成以上功能。 手动注入插件。

  1.  
    @Bean
  2.  
    public MasterSlaveAutoRoutingPlugin masterSlaveAutoRoutingPlugin(){
  3.  
    return new MasterSlaveAutoRoutingPlugin();
  4.  
    }

默认主库名称master,从库名称slave。

问题

我在配置好了之后,调试发现对数据库读的操作不得进入MasterSlaveAutoRoutingPlugin,而且进入了默认的库。只有写进入了MasterSlaveAutoRoutingPlugin中。当然也可以默认为从库,但是感觉就不是很好。

于是我自定义了一个aop切面来,来完成库的选择,代码如下:

  1.  
    import java.lang.reflect.Method;
  2.  
    import org.aspectj.lang.JoinPoint;
  3.  
    import org.aspectj.lang.annotation.After;
  4.  
    import org.aspectj.lang.annotation.Aspect;
  5.  
    import org.aspectj.lang.annotation.Before;
  6.  
    import org.aspectj.lang.annotation.Pointcut;
  7.  
    import org.aspectj.lang.reflect.MethodSignature;
  8.  
    import org.springframework.context.annotation.Lazy;
  9.  
    import org.springframework.core.annotation.Order;
  10.  
    import org.springframework.stereotype.Component;
  11.  
    import com.baomidou.dynamic.datasource.annotation.DS;
  12.  
    import com.baomidou.dynamic.datasource.toolkit.DynamicDataSourceContextHolder;
  13.  
    import lombok.extern.java.Log;
  14.  
     
  15.  
    /**
  16.  
    * Copyright: Copyright (c) 2019
  17.  
    * <p> 说明:动态数据源配置 </P>
  18.  
    *
  19.  
    * @version: V1.0
  20.  
    * @author: BianPeng
  21.  
    *
  22.  
    */
  23.  
    @Aspect
  24.  
    @Component
  25.  
    @Order(0)
  26.  
    @Lazy(false)
  27.  
    @Log
  28.  
    public class DataSourceAop{
  29.  
     
  30.  
    private static final String MASTER = "master";
  31.  
     
  32.  
    private static final String SLAVE = "slave";
  33.  
     
  34.  
     
  35.  
    @Pointcut("execution(* com.buybit.power.service..*.*(..)) || execution(* com.baomidou.mybatisplus.extension.service..*.*(..))")
  36.  
    public void checkArgs() {
  37.  
    }
  38.  
     
  39.  
    // 这里切到你的方法目录
  40.  
    @Before("checkArgs()")
  41.  
    public void process(JoinPoint joinPoint) throws NoSuchMethodException, SecurityException {
  42.  
    String methodName = joinPoint.getSignature().getName();
  43.  
    if (methodName.startsWith("get")
  44.  
    || methodName.startsWith("count")
  45.  
    || methodName.startsWith("find")
  46.  
    || methodName.startsWith("list")
  47.  
    || methodName.startsWith("select")
  48.  
    || methodName.startsWith("check")
  49.  
    || methodName.startsWith("page")) {
  50.  
     
  51.  
    log.info("当前执行的库:"+SLAVE);
  52.  
    DynamicDataSourceContextHolder.push(SLAVE);
  53.  
    } else {
  54.  
    log.info("当前执行的库:"+MASTER);
  55.  
    DynamicDataSourceContextHolder.push(MASTER);
  56.  
    }
  57.  
    }
  58.  
    @After("checkArgs()")
  59.  
    public void afterAdvice(){
  60.  
    DynamicDataSourceContextHolder.clear();
  61.  
    }
  62.  
    }

但是发现,baomidou/dynamic-datasource自带的@DS没失去了着用,于是我把有@DS的类和方法排除掉,代码入下:

  1.  
    import java.lang.reflect.Method;
  2.  
    import org.aspectj.lang.JoinPoint;
  3.  
    import org.aspectj.lang.annotation.After;
  4.  
    import org.aspectj.lang.annotation.Aspect;
  5.  
    import org.aspectj.lang.annotation.Before;
  6.  
    import org.aspectj.lang.annotation.Pointcut;
  7.  
    import org.aspectj.lang.reflect.MethodSignature;
  8.  
    import org.springframework.context.annotation.Lazy;
  9.  
    import org.springframework.core.annotation.Order;
  10.  
    import org.springframework.stereotype.Component;
  11.  
    import com.baomidou.dynamic.datasource.annotation.DS;
  12.  
    import com.baomidou.dynamic.datasource.toolkit.DynamicDataSourceContextHolder;
  13.  
    import lombok.extern.java.Log;
  14.  
     
  15.  
    /**
  16.  
    * Copyright: Copyright (c) 2019
  17.  
    * <p> 说明:动态数据源配置 </P>
  18.  
    *
  19.  
    * @version: V1.0
  20.  
    * @author: BianPeng
  21.  
    *
  22.  
    */
  23.  
    @Aspect
  24.  
    @Component
  25.  
    @Order(0)
  26.  
    @Lazy(false)
  27.  
    @Log
  28.  
    public class DataSourceAop{
  29.  
     
  30.  
    private static final String MASTER = "master";
  31.  
     
  32.  
    private static final String SLAVE = "slave";
  33.  
     
  34.  
     
  35.  
    @Pointcut("execution(* com.buybit.power.service..*.*(..)) || execution(* com.baomidou.mybatisplus.extension.service..*.*(..))")
  36.  
    public void checkArgs() {
  37.  
    }
  38.  
     
  39.  
    // 这里切到你的方法目录
  40.  
    @Before("checkArgs()")
  41.  
    public void process(JoinPoint joinPoint) throws NoSuchMethodException, SecurityException {
  42.  
    String methodName = joinPoint.getSignature().getName();
  43.  
    Class clazz = joinPoint.getTarget().getClass();
  44.  
    if(clazz.isAnnotationPresent(DS.class)){
  45.  
    //获取类上注解
  46.  
    return;
  47.  
    }
  48.  
     
  49.  
    String targetName = clazz.getSimpleName();
  50.  
    Class[] parameterTypes =
  51.  
    ((MethodSignature)joinPoint.getSignature()).getMethod().getParameterTypes();
  52.  
    Method methdo = clazz.getMethod(methodName,parameterTypes);
  53.  
    if (methdo.isAnnotationPresent(DS.class)) {
  54.  
    return;
  55.  
    }
  56.  
    if (methodName.startsWith("get")
  57.  
    || methodName.startsWith("count")
  58.  
    || methodName.startsWith("find")
  59.  
    || methodName.startsWith("list")
  60.  
    || methodName.startsWith("select")
  61.  
    || methodName.startsWith("check")
  62.  
    || methodName.startsWith("page")) {
  63.  
     
  64.  
    log.info("当前执行的库:"+SLAVE);
  65.  
    DynamicDataSourceContextHolder.push(SLAVE);
  66.  
    } else {
  67.  
    log.info("当前执行的库:"+MASTER);
  68.  
    DynamicDataSourceContextHolder.push(MASTER);
  69.  
    }
  70.  
    }
  71.  
    @After("checkArgs()")
  72.  
    public void afterAdvice(){
  73.  
    DynamicDataSourceContextHolder.clear();
  74.  
    }
  75.  
    }

这样可以让你有@DS的注解依然生效,而且也会根据方法名来自动切换数据源。

baomidou的dynamic-datasource读写分离实现和加入AOP根据方法名选择库的更多相关文章

  1. SSM/SSH框架的MySQL 读写分离实现的一种简单方法

    简介 MySQL已经是使用最为广泛的一种数据库,往往实际使用过程中,为实现高可用及高性能,项目会采用主丛复制的方式实现读写分离.MySQL本身支持复制,通过简单的配置即可实现一主多从的配置,具体实现可 ...

  2. 【EFCore框架】支持多数库 、支持读写分离框架、支持事务提交保存、跨库实现事务

    1.支持多数库: 数据库上下文实现 多数据支持: 2. 支持读写分离框架(支持多从库随机) 工作单元(以及仓储)实现读写分离 配置文件: 3. 支持事务提交保存 工作单元 实现事务数据统一提交保存 具 ...

  3. laravel(lumen)配置读写分离后,强制读主(写)库数据库,解决主从延迟问题

    在Model里面加上下面这句,强制读主(写)库数据库,解决主从延迟问题. public static function boot() { //清空从连接,会自动使用主连接 DB::connection ...

  4. Springboot + Mysql8实现读写分离

    在实际的生产环境中,为了确保数据库的稳定性,我们一般会给数据库配置双机热备机制,这样在master数据库崩溃后,slave数据库可以立即切换成主数据库,通过主从复制的方式将数据从主库同步至从库,在业务 ...

  5. 从零开始学 Java - Spring AOP 实现主从读写分离

    深刻讨论为什么要读写分离? 为了服务器承载更多的用户?提升了网站的响应速度?分摊数据库服务器的压力?就是为了双机热备又不想浪费备份服务器?上面这些回答,我认为都不是错误的,但也都不是完全正确的.「读写 ...

  6. JAVA-mysql读写分离插件介绍

    kingshard是一个由Go开发高性能MySQL Proxy项目,kingshard在满足基本的读写分离的功能上,致力于简化MySQL分库分表操作:能够让DBA通过kingshard轻松平滑地实现M ...

  7. Spring Boot2(四):使用Spring Boot多数据源实现读写分离

    前言 实际业务场景中,不可能只有一个库,所以就有了分库分表,多数据源的出现.实现了读写分离,主库负责增改删,从库负责查询.这篇文章将实现Spring Boot如何实现多数据源,动态数据源切换,读写分离 ...

  8. Mybatis多数据源读写分离(注解实现)

    #### Mybatis多数据源读写分离(注解实现) ------ 首先需要建立两个库进行测试,我这里使用的是master_test和slave_test两个库,两张库都有一张同样的表(偷懒,喜喜), ...

  9. 读写分离很难吗?springboot结合aop简单就实现了

    目录 前言 环境部署 开始项目 注意 參考: 前言 入职新公司到现在也有一个月了,完成了手头的工作,前几天终于有时间研究下公司旧项目的代码.在研究代码的过程中,发现项目里用到了Spring Aop来实 ...

  10. 原理解密 → Spring AOP 实现动态数据源(读写分离),底层原理是什么

    开心一刻 女孩睡醒玩手机,收到男孩发来一条信息:我要去跟我喜欢的人表白了! 女孩的心猛的一痛,回了条信息:去吧,祝你好运! 男孩回了句:但是我没有勇气说不来,怕被打! 女孩:没事的,我相信你!此时女孩 ...

随机推荐

  1. problemmatcher 引用无效: $esbuild-watch vscode插件报错

    vscode 插件 esbuild类型提示报错 最近在上手开发vscode插件,demo阶段就遇到了一个小问题. 搜索引擎没有特别好的回答, 记录一下,以供查漏补缺. vscode插件开发 做为一统前 ...

  2. Python入门:A+B问题

    1. A + B 问题I 前言 本篇作为课程第一篇,主要是对Python基础语法进行扫盲,本节课会学习到下面知识: 输入 本道题目的工作任务很简单,只是计算两个数的和,但是在计算之前,我们首先要明确的 ...

  3. 『玩转Streamlit』--环境配置

    尽管Streamlit的使用非常直观,但正确的环境配置对于充分发挥其潜力仍然至关重要. 本篇将介绍如何从头开始配置Streamlit环境,以及Streamlit开发过程中常用的几个命令. 最后通过一个 ...

  4. 云原生周刊:Prometheus 3.0 Beta 发布|2024.09.16

    开源项目推荐 Kuma Kuma 是一个现代化的基于 Envoy 的服务网格,能够在每个云平台上运行,支持单区域或多区域部署,兼容 Kubernetes 和虚拟机.凭借其广泛的通用工作负载支持,以及对 ...

  5. KubeSphere 宣布开源 Thanos 的企业级发行版 Whizard

    日前,青云科技宣布开源 Thanos 的企业级发行版 Whizard,为企业带来真正高可用.可扩展.可存储与查询海量监控数据.易运维.安全的 Prometheus 长期存储方案. Prometheus ...

  6. 云原生周刊:Argo CD v2.12 发布候选版本 | 2024.06.24

    开源项目推荐 kOps kOps - Kubernetes Operations,是一个用于创建.销毁.升级和维护生产级高可用 Kubernetes 集群的工具.它支持在 AWS.GCP.Digita ...

  7. burpsuit无法启动

    在安装burpsuit时,最难的并不是找带有注册机的burpsuit安装包. 而是因为一些底层的原因,无法打开 burpsuit. 提示 Your JRE appears to be version ...

  8. 关于C语言指针类型的总结

    前言 我个人将目前在C语言中所遇到的指针归类为8种,至于为何写第九点,是因为我个人认为第九点极容易与第五点混淆,故总结如下: 1.普通指针 普通指针即最常见的如:int * . char*等 甚至于也 ...

  9. 初探python栈帧逃逸

    前言 以前在一些大型比赛就遇到这种题,一直没时间去研究,现在康复训练下:) 生成器介绍 生成器(Generator)是Python中一种特殊的迭代器,它可以在迭代过程中动态生成值,而不需要一次性将所有 ...

  10. SSIS错误汇总

    问题一 [Excel 目标 [2]] 错误: SSIS 错误代码 DTS_E_CANNOTACQUIRECONNECTIONFROMCONNECTIONMANAGER.对连接管理器"未匹配用 ...