锋盈数科-知识库 Logo
首页
软件开发
计算机基础
Hello Halo
新手必读
关于本知识库
登录 →
锋盈数科-知识库 Logo
首页 软件开发 计算机基础 Hello Halo 新手必读 关于本知识库
登录
  1. 首页
  2. 软件开发
  3. JAVA
  4. Spring Boot整合MyBatis Plus集成多数据源轻松实现数据读写分离

Spring Boot整合MyBatis Plus集成多数据源轻松实现数据读写分离

0
  • JAVA
  • 发布于 2024-08-14
  • 0 次阅读
黄健
黄健

本文由 简悦 SimpRead 转码, 原文地址 blog.csdn.net

文章目录

    • 1、前言
    • 2、配置多数据源
    • 3、创建数据源和数据库连接池 Bean
    • 4、配置 MyBatis Plus 数据源
    • 5、配置动态数据源路由
    • 6、配置数据源切换注解
    • 7、编写 Service 层代码
    • 8、总结

1、前言

在实际项目中,数据库是至关重要的组成部分。为了提高性能和可用性,常见的数据库优化策略之一是将数据库读和写操作分离,以降低数据库服务器的负载。

MyBatis Plus 是一个流行的 Java 持久化框架,它提供了读写分离的支持,结合 Spring Boot 集成多数据源的特性可以帮助你轻松地实现读写分离策略。

Springboot 项目中操作数据库,使用 MP 方便快捷;如何使用 MP 实现读写分离,以提高你的应用程序性能和可用性。我们将分为以下几个步骤来介绍这一过程。

2、配置多数据源

在 Spring Boot 配置文件中配置多个数据源,通常至少包括一个主数据库(写库)和一个或多个从数据库(读库)。以下是一个示例的配置:

spring.datasource.write.url=jdbc:mysql://write-server:3306/write_db
spring.datasource.write.username=root
spring.datasource.write.password=root
spring.datasource.read.url=jdbc:mysql://read1-server:3306/read1_db
spring.datasource.read.username=root
spring.datasource.read.password=root

上面配置中,write 数据源代表主数据库,read 数据源代表从数据库。你可以根据项目需求配置更多的从数据库。

3、创建数据源和数据库连接池 Bean

在 Spring Boot 配置中,为每个数据源创建一个数据源和数据库连接池 Bean。

@Bean(name = "writeDataSource")
@ConfigurationProperties(prefix = "spring.datasource.write")
public DataSource writeDataSource() {
     return DataSourceBuilder.create().build();
}

@Bean(name = "readDataSource")
@ConfigurationProperties(prefix = "spring.datasource.read")
public DataSource readDataSource() {
   return DataSourceBuilder.create().build();
}

4、配置 MyBatis Plus 数据源

配置 MyBatis Plus 的数据源和 SqlSessionFactoryBean,分别指向主数据源和从数据源。

@Bean(name = "writeSqlSessionFactory")
public SqlSessionFactory writeSqlSessionFactory(@Qualifier("writeDataSource") DataSource writeDataSource) throws Exception {
    MybatisSqlSessionFactoryBean sessionFactoryBean = new MybatisSqlSessionFactoryBean();
   sessionFactoryBean.setDataSource(writeDataSource);
      // 配置 MyBatis Plus的一些参数
     return sessionFactoryBean.getObject();
}

@Bean(name = "readSqlSessionFactory")
public SqlSessionFactory readSqlSessionFactory(@Qualifier("readDataSource") DataSource read1DataSource) throws Exception {
   MybatisSqlSessionFactoryBean sessionFactoryBean = new MybatisSqlSessionFactoryBean();
   sessionFactoryBean.setDataSource(readDataSource);
  // 配置 MyBatis Plus的一些参数
   return sessionFactoryBean.getObject();
}

5、配置动态数据源路由

创建一个动态数据源路由,用于根据操作类型(读操作或写操作)选择主数据源或从数据源。

public class DynamicDataSource extends AbstractRoutingDataSource {
   @Override
   protected Object determineCurrentLookupKey() {
       return DataSourceContextHolder.getDataSource();
   }
}

6、配置数据源切换注解

创建自定义注解(如 @ReadDataSource 和 @WriteDataSource),并创建拦截器,根据注解来选择数据源。

首先,创建 @ReadDataSource 注解:

import java.lang.annotation.*;
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface ReadDataSource {
}

接下来,创建 @WriteDataSource 注解:

import java.lang.annotation.*;
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface WriteDataSource {
}

然后,创建数据源切换拦截器,该拦截器将根据注解选择数据源:

import org.aspectj.lang.JoinPoint;
 import org.aspectj.lang.annotation.Aspect;
 import org.aspectj.lang.annotation.Before;
 import org.springframework.core.annotation.AnnotationUtils;
 import org.springframework.stereotype.Component;
 
 @Aspect
 @Component
 public class DataSourceSwitchAspect {
    @Before("execution(* com.example.service.*.*(..))")
    public void switchDataSource(JoinPoint joinPoint) {
        Class<?> targetClass = joinPoint.getTarget().getClass();
        // 检查方法上是否有数据源注解
        ReadDataSource readDataSource = AnnotationUtils.findAnnotation(targetClass, ReadDataSource.class);
        WriteDataSource writeDataSource = AnnotationUtils.findAnnotation(targetClass, WriteDataSource.class);

        if (readDataSource != null) {
            DataSourceContextHolder.setDataSource(DataSourceType.READ);
        } else if (writeDataSource != null) {
            DataSourceContextHolder.setDataSource(DataSourceType.WRITE);
        } else {
            // 没有注解,默认使用写数据源
            DataSourceContextHolder.setDataSource(DataSourceType.WRITE);
        }
    }
}

这个拦截器会在方法执行前根据注解来选择数据源,如果方法上有 @ReadDataSource 注解,则选择读数据源,如果方法上有 @WriteDataSource 注解,则选择写数据源,否则默认使用写数据源。

最后,创建一个数据源上下文持有器 DataSourceContextHolder,用于存储当前选择的数据源:

public class DataSourceContextHolder {
  private static final ThreadLocal<DataSourceType> contextHolder = new ThreadLocal<>();
  public static void setDataSource(DataSourceType dataSourceType) {
    contextHolder.set(dataSourceType);
  }

  public static DataSourceType getDataSource() {
    return contextHolder.get();
  }

  public static void clearDataSource() {
    contextHolder.remove();
  }
}

DataSourceType 是一个自定义的枚举,用于表示数据源类型(读或写):

public enum DataSourceType {
    READ, WRITE
}

通过以上配置,你可以在 Service 层的方法上使用 @ReadDataSource 和 @WriteDataSource 注解来标识读操作和写操作,拦截器会根据注解选择相应的数据源。

7、编写 Service 层代码

在 Service 层的方法中,根据操作类型(读操作或写操作),选择数据源。可以使用自定义注解或其他方式标识读操作和写操作的方法。

@Service
public class UserService {
  @Autowired
  private UserService userService; // MyBatis Plus 提供的 Service

  @ReadDataSource // 自定义的注解,用于标识读操作
  public User getUserById(Long id) {
    return userService.getById(id); // 直接使用 MyBatis Plus 提供的查询方法
  }

  @WriteDataSource // 自定义的注解,用于标识写操作
  public void createUser(User user) {
    userService.save(user); // 直接使用 MyBatis Plus 提供的插入方法
  }
}

8、总结

MyBatis Plus 的读写分离功能使数据库优化变得轻而易举。通过将读和写操作分离到不同的数据源中,你可以降低数据库负载,提高应用程序性能和可用性。需要注意的是,具体的实现方式和注解可以根据项目需求和业务逻辑进行调整和扩展。

已经是最低谷了,怎么走都是向上,所以,你怕什么。

标签: #软件开发 1171 #JAVA 991
相关文章

Spring 实现 3 种异步接口 2024-10-18 09:07

大家好,我是苏三~ 如何处理比较耗时的接口? 这题我熟,直接上异步接口,使用 Callable、WebAsyncTask 和 DeferredResult、CompletableFuture等均可实现。 但这些方法有局限性,处理结果仅返回单个值。在某些场景下,如果需要接口异步处理的同时,还持续不断地

重学SpringBoot3-集成Redis(五)之布隆过滤器 2024-10-08 11:24

更多SpringBoot3内容请关注我的专栏:《SpringBoot3》 期待您的点赞👍收藏⭐评论✍ 重学SpringBoot3-集成Redis(五)之布隆过滤器 1. 什么是布隆过滤器? * 基本概念 适用场景 2. 使用 Redis 实现布隆过滤器 * 项目依赖 Redis 配置

SpringBoot整合异步任务执行 2024-10-08 11:24

同步任务: 同步任务是在单线程中按顺序执行,每次只有一个任务在执行,不会引发线程安全和数据一致性等 并发问题 同步任务需要等待任务执行完成后才能执行下一个任务,无法同时处理多个任务,响应慢,影响用 户体验 异步任务: 异步任务是在多线程中同时执行,多个任务可以并发执行,同时处理多个请求,响应快,资源

springboot kafka多数据源,通过配置动态加载发送者和消费者 2024-10-08 11:24

前言 最近做项目,需要支持kafka多数据源,实际上我们也可以通过代码固定写死多套kafka集群逻辑,但是如果需要不修改代码扩展呢,因为kafka本身不处理额外逻辑,只是起到削峰,和数据的传递,那么就需要对架构做一定的设计了。 准备test kafka本身非常容易上手,如果我们需要单元测试,引入ja

SpringBoot 集成 Redis 2024-10-08 11:24

一:SpringBoot 集成 Redis ①Redis是一个 NoSQL(not only)数据库, 常作用缓存 Cache 使用。 ②Redis是一个中间件、是一个独立的服务器;常用的数据类型: string , hash ,set ,zset , list ③通过Redis客户端可以使用多种语

SpringBoot整合QQ邮箱 2024-10-08 11:24

SpringBoot可以通过导入依赖的方式集成多种技术,这当然少不了我们常用的邮箱,现在本章演示SpringBoot整合QQ邮箱发送邮件…. 下面按步骤进行: 1.获取QQ邮箱授权码 1.1 登录QQ邮箱 1.2 开启SMTP服务 找到下图中的SMTP服务区域,如果当前账号未开启的话自己手动开启。

目录

IT 外包服务商

  • 意见投递
  • zyf6619

软件开发应用

主菜单

  • 首页
  • 软件开发
  • 计算机基础
  • Hello Halo
  • 新手必读
  • 关于本知识库
Copyright © 2024 your company All Rights Reserved. Powered by Halo.