SpringBoot+MyBatis-Plus實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫分離的代碼示例
1. 引言
在當(dāng)今互聯(lián)網(wǎng)應(yīng)用中,數(shù)據(jù)庫(kù)讀寫分離是提高系統(tǒng)性能和穩(wěn)定性的重要手段之一。通過(guò)將讀操作和寫操作分別路由到不同的數(shù)據(jù)庫(kù)節(jié)點(diǎn),可以有效減輕數(shù)據(jù)庫(kù)服務(wù)器的負(fù)擔(dān),提升系統(tǒng)的整體性能。本文將介紹如何利用Spring Boot和MyBatis-Plus框架實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫分離,并通過(guò)簡(jiǎn)單易懂的代碼示例來(lái)詳細(xì)說(shuō)明每個(gè)步驟。
2. MyBatis-Plus簡(jiǎn)介
MyBatis-Plus是MyBatis的增強(qiáng)工具,提供了許多實(shí)用的功能,包括但不限于代碼生成器、通用Mapper、分頁(yè)插件等。在本文中,我們將專注于使用MyBatis-Plus實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫分離。
3. 準(zhǔn)備工作
在開始之前,確保你的開發(fā)環(huán)境中已經(jīng)安裝了以下軟件:
- JDK(推薦使用JDK 8及以上版本)
- Maven
- IntelliJ IDEA或Eclipse(可選)
在項(xiàng)目的pom.xml
文件中添加MyBatis-Plus和數(shù)據(jù)庫(kù)驅(qū)動(dòng)的依賴:
<!-- MyBatis-Plus --> <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.4.3.2</version> </dependency> <!-- MySQL驅(qū)動(dòng) --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.26</version> </dependency>
4. 配置數(shù)據(jù)源
在application.properties
(或application.yml
)中配置主庫(kù)和從庫(kù)的數(shù)據(jù)源:
# 主庫(kù)數(shù)據(jù)源配置 spring.datasource.master.url=jdbc:mysql://localhost:3306/masterdb?useSSL=false&serverTimezone=UTC spring.datasource.master.username=root spring.datasource.master.password=root spring.datasource.master.driver-class-name=com.mysql.cj.jdbc.Driver # 從庫(kù)數(shù)據(jù)源配置 spring.datasource.slave.url=jdbc:mysql://localhost:3307/slavedb?useSSL=false&serverTimezone=UTC spring.datasource.slave.username=root spring.datasource.slave.password=root spring.datasource.slave.driver-class-name=com.mysql.cj.jdbc.Driver
5. 配置MyBatis-Plus
創(chuàng)建一個(gè)配置類,用于配置MyBatis-Plus的分頁(yè)插件和動(dòng)態(tài)數(shù)據(jù)源:
import com.baomidou.mybatisplus.extension.plugins.PaginationInterceptor; import com.baomidou.mybatisplus.extension.spring.MybatisSqlSessionFactoryBean; import org.apache.ibatis.session.SqlSessionFactory; import org.mybatis.spring.annotation.MapperScan; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.jdbc.datasource.DataSourceTransactionManager; import org.springframework.transaction.PlatformTransactionManager; import javax.sql.DataSource; @Configuration @MapperScan(basePackages = "com.example.demo.mapper", sqlSessionFactoryRef = "sqlSessionFactory") public class MybatisPlusConfig { @Bean(name = "sqlSessionFactory") public SqlSessionFactory sqlSessionFactory(@Qualifier("dataSource") DataSource dataSource) throws Exception { MybatisSqlSessionFactoryBean sqlSessionFactoryBean = new MybatisSqlSessionFactoryBean(); sqlSessionFactoryBean.setDataSource(dataSource); return sqlSessionFactoryBean.getObject(); } @Bean(name = "transactionManager") public PlatformTransactionManager transactionManager(@Qualifier("dataSource") DataSource dataSource) { return new DataSourceTransactionManager(dataSource); } @Bean public PaginationInterceptor paginationInterceptor() { return new PaginationInterceptor(); } }
6. 創(chuàng)建實(shí)體類和Mapper接口
創(chuàng)建一個(gè)簡(jiǎn)單的實(shí)體類和對(duì)應(yīng)的Mapper接口:
// User.java import com.baomidou.mybatisplus.annotation.TableId; import com.baomidou.mybatisplus.annotation.TableName; import lombok.Data; @Data @TableName("user") public class User { @TableId private Long id; private String username; private String password; }
// UserMapper.java import com.baomidou.mybatisplus.core.mapper.BaseMapper; import org.apache.ibatis.annotations.Mapper; @Mapper public interface UserMapper extends BaseMapper<User> { }
7. 編寫Service
創(chuàng)建Service層,調(diào)用Mapper接口完成數(shù)據(jù)操作:
// UserService.java import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import org.springframework.stereotype.Service; @Service public class UserService extends ServiceImpl<UserMapper, User> { }
8. 控制器層
編寫Controller層,暴露接口供前端調(diào)用:
// UserController.java import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; @RestController @RequestMapping("/users") public class UserController { private final UserService userService; @Autowired public UserController(UserService userService) { this.userService = userService; } @GetMapping("/{id}") public User getUser(@PathVariable Long id) { return userService.getById(id); } @PostMapping public boolean createUser(@RequestBody User user) { return userService.save(user); } @PutMapping public boolean updateUser(@RequestBody User user) { return userService.updateById(user); } @DeleteMapping("/{id}") public boolean deleteUser(@PathVariable Long id) { return userService.removeById(id); } }
9. 測(cè)試
啟動(dòng)Spring Boot應(yīng)用程序,訪問(wèn)相應(yīng)的接口進(jìn)行測(cè)試。通過(guò)日志可以看到,MyBatis-Plus會(huì)在執(zhí)行查詢時(shí)根據(jù)一定的規(guī)則選擇主庫(kù)或從庫(kù)。
10. 數(shù)據(jù)庫(kù)讀寫分離的原理
數(shù)據(jù)庫(kù)讀寫分離的實(shí)現(xiàn)原理主要通過(guò)在MyBatis-Plus中使用@DataSource
注解,根據(jù)不同的操作選擇不同的數(shù)據(jù)源。這里簡(jiǎn)要說(shuō)明一下原理:
創(chuàng)建多個(gè)數(shù)據(jù)源: 配置文件中定義了主庫(kù)和從庫(kù)兩個(gè)數(shù)據(jù)源。
配置動(dòng)態(tài)數(shù)據(jù)源: 在
MybatisPlusConfig
配置類中,使用DynamicDataSource
類包裝主庫(kù)和從庫(kù)的數(shù)據(jù)源,通過(guò)@Primary
注解標(biāo)識(shí)主庫(kù)。自定義注解: 創(chuàng)建
@DataSource
注解,用于標(biāo)識(shí)Mapper方法應(yīng)該使用哪個(gè)數(shù)據(jù)源。AOP切面: 利用AOP,在Mapper方法執(zhí)行前根據(jù)
@DataSource
注解的值動(dòng)態(tài)切換數(shù)據(jù)源。
11. 拓展
11.1. 動(dòng)態(tài)數(shù)據(jù)源
切換策略
在實(shí)際應(yīng)用中,動(dòng)態(tài)數(shù)據(jù)源切換的策略可以根據(jù)業(yè)務(wù)需求來(lái)定制。可以基于用戶的讀寫操作比例、數(shù)據(jù)庫(kù)實(shí)例的性能等因素,靈活調(diào)整數(shù)據(jù)源切換的策略。
11.2. 多數(shù)據(jù)源事務(wù)管理
當(dāng)涉及到跨數(shù)據(jù)源的事務(wù)時(shí),需要謹(jǐn)慎處理??梢酝ㄟ^(guò)使用分布式事務(wù)框架(如Seata、TCC事務(wù)等)來(lái)保障事務(wù)的一致性。
11.3. 多租戶支持
在一些場(chǎng)景中,需要為不同的租戶提供獨(dú)立的數(shù)據(jù)庫(kù),此時(shí)可以考慮使用多租戶架構(gòu),并根據(jù)租戶信息動(dòng)態(tài)切換數(shù)據(jù)源。
12. 總結(jié)
通過(guò)本文的學(xué)習(xí),我們了解了如何利用Spring Boot和MyBatis-Plus實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫分離。這一策略在提升系統(tǒng)性能和穩(wěn)定性方面有著顯著的效果。通過(guò)合理配置數(shù)據(jù)源、使用MyBatis-Plus框架以及編寫自定義注解和AOP切面,我們成功地搭建了一個(gè)簡(jiǎn)單而完整的讀寫分離系統(tǒng)。
希望這篇文章對(duì)你理解Spring Boot和MyBatis-Plus的讀寫分離實(shí)現(xiàn)提供了幫助。在實(shí)際項(xiàng)目中,可以根據(jù)具體的業(yè)務(wù)需求和性能要求調(diào)整和拓展這一方案。
以上就是SpringBoot+MyBatis-Plus實(shí)現(xiàn)數(shù)據(jù)庫(kù)讀寫分離的代碼示例的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot+MyBatis-Plus讀寫分離的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
java中的通用權(quán)限管理設(shè)計(jì)(推薦)
下面小編就為大家推薦一篇java中的通用權(quán)限管理設(shè)計(jì),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2017-12-12Java實(shí)現(xiàn)多數(shù)據(jù)源的幾種方式總結(jié)
這篇文章主要給大家總結(jié)介紹了關(guān)于Java實(shí)現(xiàn)多數(shù)據(jù)源的幾種方式,最近項(xiàng)目中的工作流需要查詢多個(gè)數(shù)據(jù)源的數(shù)據(jù),數(shù)據(jù)源可能是不同種類的,需要的朋友可以參考下2023-08-08SpringBoot 過(guò)濾器、攔截器、監(jiān)聽器對(duì)比及使用場(chǎng)景分析
過(guò)濾器是處于客戶端和服務(wù)器資源文件之間的一道過(guò)濾網(wǎng),這篇文章主要介紹了SpringBoot 過(guò)濾器、攔截器、監(jiān)聽器對(duì)比及使用場(chǎng)景分析,需要的朋友可以參考下2021-05-05spring boot中的條件裝配bean的實(shí)現(xiàn)
這篇文章主要介紹了spring boot中的條件裝配bean的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12SpringFramework應(yīng)用接入Apollo配置中心過(guò)程解析
這篇文章主要介紹了SpringFramework應(yīng)用接入Apollo配置中心過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03Spring Mybatis 基本使用過(guò)程(推薦)
Mybatis是一個(gè)半自動(dòng)ORM(Object Relational Mapping)框架,它可以簡(jiǎn)化數(shù)據(jù)庫(kù)編程,讓開發(fā)者更專注于SQL本身,本文給大家介紹Spring Mybatis 基本使用過(guò)程,感興趣的朋友跟隨小編一起看看吧2024-09-09SpringBoot接口請(qǐng)求入?yún)⒑统鰠⒃鰪?qiáng)的五種方法
這篇文章主要介紹了SpringBoot接口請(qǐng)求入?yún)⒑统鰠⒃鰪?qiáng)的五種方法,使用`@JsonSerialize`和`@JsonDeserialize`注解,全局配置Jackson的`ObjectMapper`,使用`@ControllerAdvice`配合`@InitBinder`,自定義HttpMessageConverter和使用AOP進(jìn)行切面編程,需要的朋友可以參考下2024-07-07