前言

随着业务量的不断增长,数据库的读写压力也越来越大。为了解决这个问题,我们可以采用读写分离的方案来分担数据库的读写负载。本文将介绍如何使用 Spring Boot + MyBatis Plus + MySQL 实现读写分离。

读写分离原理

读写分离是指将数据库的读操作和写操作分别放到不同的数据库实例上,从而达到分担数据库负载的目的。一般情况下,写操作的频率比读操作高,因此可以将写操作放到主库上,将读操作放到从库上。这样可以保证主库的写入性能,同时也可以提高从库的读取性能。

实现步骤

1. 主从复制搭建

首先,我们需要创建两个数据库实例,一个用于主库,一个用于从库。这里我们使用 MySQL 数据库作为示例。

参见搭建mysql主从复制一文。

2.配置pom.xml

  1. <dependency>
  2. <groupId>mysql</groupId>
  3. <artifactId>mysql-connector-java</artifactId>
  4. <scope>runtime</scope>
  5. </dependency>
  6. <dependency>
  7. <groupId>com.baomidou</groupId>
  8. <artifactId>mybatis-plus-boot-starter</artifactId>
  9. </dependency>
  10. <!-- 连接池 -->
  11. <dependency>
  12. <groupId>com.alibaba</groupId>
  13. <artifactId>druid-spring-boot-starter</artifactId>
  14. <version>1.1.9</version>
  15. </dependency>
  16. <!--aop -->
  17. <dependency>
  18. <groupId>org.springframework.boot</groupId>
  19. <artifactId>spring-boot-starter-aop</artifactId>
  20. </dependency>
  21. <dependency>
  22. <groupId>org.aspectj</groupId>
  23. <artifactId>aspectjtools</artifactId>
  24. <version>1.8.13</version>
  25. </dependency>

3. 配置数据源

在 Spring Boot 中,我们可以使用 application.yml 文件来配置数据源。在这里,我们需要配置两个数据源,一个用于主库,一个用于从库。具体配置如下:

  1. mysql:
  2. datasource:
  3. readNum: 1
  4. type: com.alibaba.druid.pool.DruidDataSource
  5. write:
  6. username: root
  7. password: 123456
  8. driver-class-name: com.mysql.cj.jdbc.Driver
  9. url: jdbc:mysql://10.10.26.212:3306/test?useSSL=false&useUnicode=true
  10. minIdle: 5
  11. maxActive: 100
  12. initialSize: 10
  13. maxWait: 60000
  14. timeBetweenEvictionRunsMillis: 60000
  15. minEvictableIdleTimeMillis: 300000
  16. validationQuery: select 'x'
  17. testWhileIdle: true
  18. testOnBorrow: false
  19. testOnReturn: false
  20. poolPreparedStatements: true
  21. maxPoolPreparedStatementPerConnectionSize: 50
  22. removeAbandoned: true
  23. filters: stat
  24. read1:
  25. username: root
  26. password: 123456
  27. driver-class-name: com.mysql.cj.jdbc.Driver
  28. url: jdbc:mysql://10.10.26.213:3306/test?useSSL=false&useUnicode=true
  29. minIdle: 5
  30. maxActive: 100
  31. initialSize: 10
  32. maxWait: 60000
  33. timeBetweenEvictionRunsMillis: 60000
  34. minEvictableIdleTimeMillis: 300000
  35. validationQuery: select 'x'
  36. testWhileIdle: true
  37. testOnBorrow: false
  38. testOnReturn: false
  39. poolPreparedStatements: true
  40. maxPoolPreparedStatementPerConnectionSize: 50
  41. removeAbandoned: true
  42. filters: stat

4. 配置 MyBatis Plus

MyBatis Plus 是一个 MyBatis 的增强工具,它可以简化 MyBatis 的开发流程。在这里,我们需要在 application.yml 文件中配置 MyBatis Plus 的相关参数。具体配置如下:

  1. mybatis-plus:
  2. mapper-locations: classpath:mapper/*.xml
  3. configuration:
  4. map-underscore-to-camel-case: true
  5. cache-enabled: false
  6. typeAliasesPackage: com.sandy.dyds.model

5. 实现读写分离

首先,我们需要创建一个 DbContextHolder 类,用于保存当前线程使用的数据源。具体实现如下:

  1. @Log4j2
  2. public class DbContextHolder {
  3. public static final String WRITE = "write";
  4. public static final String READ = "read";
  5. private static ThreadLocal<String> contextHolder = new ThreadLocal<>();
  6. public static void setDbType(String dbType) {
  7. if(dbType == null) {
  8. throw new NullPointerException();
  9. }
  10. contextHolder.set(dbType);
  11. }
  12. public static String getDbType() {
  13. return contextHolder.get() == null ? WRITE : contextHolder.get();
  14. }
  15. public static void clearDbType() {
  16. contextHolder.remove();
  17. }
  18. }

然后,我们需要创建一个 RoutingDataSource 类,用于动态切换数据源。具体实现如下:

  1. @Log4j2
  2. public class RoutingDataSource extends AbstractRoutingDataSource {
  3. @Value("${mysql.datasource.readNum}")
  4. private int num;
  5. @Override
  6. protected Object determineCurrentLookupKey() {
  7. String typeKey = DbContextHolder.getDbType();
  8. if(typeKey.equals(DbContextHolder.WRITE)) {
  9. return typeKey;
  10. }
  11. //使用随机数决定使用哪个读库
  12. //在1-N之间生成整型随机数
  13. int random = (int) (Math.random() * 1) + num;
  14. return DbContextHolder.READ + random;
  15. }
  16. }

接着,我们需要创建一个 DataSourceConfig 类,用于配置数据源。具体实现如下:

  1. @Configuration
  2. public class DataSourceConfig {
  3. @Value("${mysql.datasource.type}")
  4. private Class<? extends DataSource> dataSourceType;
  5. /**
  6. * 写数据源
  7. * Primary 标志这个 Bean 如果在多个同类 Bean 候选时,该 Bean 优先被考虑。
  8. * 多数据源配置的时候注意,必须要有一个主数据源,用 @Primary 标志该 Bean
  9. */
  10. @Primary
  11. @Bean
  12. @ConfigurationProperties(prefix = "mysql.datasource.write")
  13. public DataSource writeDataSource() {
  14. DataSource ds = DataSourceBuilder.create().type(dataSourceType).build();
  15. return ds;
  16. }
  17. @Bean
  18. @ConfigurationProperties(prefix = "mysql.datasource.read1")
  19. public DataSource readDataSource_1() {
  20. DataSource ds = DataSourceBuilder.create().type(dataSourceType).build();
  21. return ds;
  22. }
  23. /**
  24. * 设置数据源路由,通过该类中的determineCurrentLookupKey决定使用哪个数据源
  25. */
  26. @Bean
  27. public AbstractRoutingDataSource routingDataSource() {
  28. RoutingDataSource proxy = new RoutingDataSource();
  29. Map<Object, Object> targetDataSources = new HashMap<>();
  30. targetDataSources.put(DbContextHolder.WRITE, writeDataSource());
  31. targetDataSources.put(DbContextHolder.READ + "1", readDataSource_1());
  32. proxy.setDefaultTargetDataSource(writeDataSource());
  33. proxy.setTargetDataSources(targetDataSources);
  34. return proxy;
  35. }
  36. /**
  37. * 由于Spring容器中现在有多个数据源,所以我们需要为事务管理器和MyBatis手动指定一个明确的数据源。
  38. */
  39. @Bean
  40. public SqlSessionFactory sqlSessionFactory() throws Exception {
  41. MybatisSqlSessionFactoryBean sqlSessionFactory = new MybatisSqlSessionFactoryBean();
  42. sqlSessionFactory.setDataSource(routingDataSource());
  43. MybatisConfiguration configuration = new MybatisConfiguration();
  44. configuration.setJdbcTypeForNull(JdbcType.NULL);
  45. configuration.setMapUnderscoreToCamelCase(true);
  46. configuration.setCacheEnabled(false);
  47. sqlSessionFactory.setConfiguration(configuration);
  48. return sqlSessionFactory.getObject();
  49. }
  50. @Bean
  51. public DataSourceTransactionManager transactionManager() {
  52. return new DataSourceTransactionManager(routingDataSource());
  53. }
  54. }

最后,我们需要使用AOP以注解方式切换只读数据库。具体实现如下:

  1. @Target({ElementType.METHOD, ElementType.TYPE})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. public @interface ReadOnly {
  4. }
  1. @Aspect
  2. @Component
  3. @Log4j2
  4. public class DataSourceAop implements Ordered {
  5. @Around("@annotation(readOnly)")
  6. public Object setRead(ProceedingJoinPoint joinPoint, ReadOnly readOnly) throws Throwable{
  7. try {
  8. DbContextHolder.setDbType(DbContextHolder.READ);
  9. return joinPoint.proceed();
  10. } finally {
  11. //清除DbType一方面为了避免内存泄漏,更重要的是避免对后续在本线程上执行的操作产生影响
  12. DbContextHolder.clearDbType();
  13. log.info("清除threadLocal");
  14. }
  15. }
  16. @Override
  17. public int getOrder() {
  18. return 0;
  19. }
  20. }

总结

通过本文的介绍,我们了解了如何使用 Spring Boot + MyBatis Plus + MySQL 实现读写分离。读写分离可以有效地分担数据库的读写负载,提高数据库的性能和可用性。希望本文能对读写分离的实现有所帮助。

SpringBoot+MyBatisPlus实现读写分离的更多相关文章

  1. SpringBoot使用Sharding-JDBC读写分离

    本文介绍SpringBoot使用当当Sharding-JDBC进行读写分离. 1.有关Sharding-JDBC 本文还是基于当当网Sharding-Jdbc的依赖,与上一篇使用Sharding-Jd ...

  2. SpringBoot+MyBatis+MySQL读写分离

    1.  引言 读写分离要做的事情就是对于一条SQL该选择哪个数据库去执行,至于谁来做选择数据库这件事儿,无非两个,要么中间件帮我们做,要么程序自己做.因此,一般来讲,读写分离有两种实现方式.第一种是依 ...

  3. SpringBoot+MyBatis+MySQL读写分离(实例)

    ​ 1. 引言 读写分离要做的事情就是对于一条SQL该选择哪个数据库去执行,至于谁来做选择数据库这件事儿,无非两个,要么中间件帮我们做,要么程序自己做.因此,一般来讲,读写分离有两种实现方式.第一种是 ...

  4. Springboot + Mysql8实现读写分离

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

  5. SpringBoot + MyBatis + MySQL 读写分离实战

    1. 引言 读写分离要做的事情就是对于一条SQL该选择哪个数据库去执行,至于谁来做选择数据库这件事儿,无非两个,要么中间件帮我们做,要么程序自己做.因此,一般来讲,读写分离有两种实现方式.第一种是依靠 ...

  6. SpringBoot 玩转读写分离

    环境概览 前言介绍 Sharding-JDBC是当当网的一个开源项目,只需引入jar即可轻松实现读写分离与分库分表.与MyCat不同的是,Sharding-JDBC致力于提供轻量级的服务框架,无需额外 ...

  7. 分库分表(3) ---SpringBoot + ShardingSphere 实现读写分离

    分库分表(3)---ShardingSphere实现读写分离 有关ShardingSphere概念前面写了两篇博客: 1.分库分表(1) --- 理论 2. 分库分表(2) --- ShardingS ...

  8. 分库分表(6)--- SpringBoot+ShardingSphere实现分表+ 读写分离

    分库分表(6)--- ShardingSphere实现分表+ 读写分离 有关分库分表前面写了五篇博客: 1.分库分表(1) --- 理论 2.分库分表(2) --- ShardingSphere(理论 ...

  9. 分库分表(7)--- SpringBoot+ShardingSphere实现分库分表 + 读写分离

    分库分表(7)--- ShardingSphere实现分库分表+读写分离 有关分库分表前面写了六篇博客: 1.分库分表(1) --- 理论 2.分库分表(2) --- ShardingSphere(理 ...

  10. Sharding-JDBC基本使用,整合Springboot实现分库分表,读写分离

    结合上一篇docker部署的mysql主从, 本篇主要讲解SpringBoot项目结合Sharding-JDBC如何实现分库分表.读写分离. 一.Sharding-JDBC介绍 1.这里引用官网上的介 ...

随机推荐

  1. 聊聊Spring扩展点BeanPostProcessor和BeanFactoryPostProcessor

    介绍 今天聊一聊spring中很重要的两个扩展点BeanPostProcessor和BeanFactoryPostProcessor,spring之所以如次强大,是因为它提供了丰富的功能给我们使用,但 ...

  2. JavaScript中计时器requestAnimationFrame、setTimeout、setInterval、setImmediate的使用和区别

    在JavaScript中,我们经常使用requestAnimationFrame.setTimeout.setInterval和setImmediate来控制代码的执行时机.它们各有特点和适用场景: ...

  3. Let's Encrypt 泛域名证书申请

    泛域名 泛域名证书又名通配符证书是SSL证书中的其中一种形式,一般会以通配符的形式(如:*.domain.com)来指定证书所要保护的域名. OV证书和DV证书都会有通配符的域名形式提供,而EV证书一 ...

  4. python获取本地ip地址1

    import socket def get_host_ip(): """ 查询本机ip地址 return: ip """ try: s = ...

  5. 【Note】(坑)一些组合恒等式的实际意义理解(和待填坑的组合数学知识)

    目录 排列组合 恒等式 (1) \(C_n^m=C_n^{n-m}\) (2) \(A_n^m+mA_n^{m-1}=A_{n+1}^m\) (3) \(C_n^{m-1}+C_n^{m}=C_{n+ ...

  6. LeeCode 90双周赛复盘

    T1: 差值数组不同的字符串 思路:数组遍历 若前两个字符串差值数组不同,则只需要继续计算第三个字符串的差值数组即可得到答案 若前两个字符串差值数组相同,则依次遍历后续字符串,直至找到不同的差值数组 ...

  7. day118:MoFang:根据激活/未激活的状态分别显示树桩&种植植物&解锁树桩&化肥/修剪/浇水/宠物粮小图标数字的显示

    登录 1.根据激活状态和未激活状态分别显示树桩 2.用户使用植物道具进行果树种植 3.解锁树桩 4.化肥/修剪/浇水/宠物粮小图标显示 种植栏的功能实现 1. 客户端需要的植物相关参数: 总树桩数量, ...

  8. 【Spring5】JdbcTemplate

    JdbcTemplate实现对数据库增删改查 步骤 导入Jar包 mysql-connector-java-8.0.28.jar:mysql数据库连接的相关依赖 spring-tx-5.2.6.REL ...

  9. Android Studio 样式和主题背景

    样式和主题背景 转载自   Styles and Themes  |  Android Developers 借助 Android 中的样式和主题背景,您可以将应用设计的细节与界面的结构和行为分开,其 ...

  10. C# 获取指定窗口的上层窗口

    如何获取当前窗口层级上方的所有窗口信息 User32有函数GetWindow function (winuser.h) - Win32 apps | Microsoft Docs,可以根据已知窗口句柄 ...