SpringBoot+Mybatis-Plus配置多數據源及數據庫的切換

當前大環境的影響下,多數據源的情況已經非常常見了,這裏記錄下怎樣實現
在網上着落一大圈,配置多數據源的方法主要有兩種:

  1. 構建多套環境,優勢是方便控制也容易集成一些簡單的分佈式事務,缺點是非動態同時代碼量較多,配置難度大。
  2. 基於spring提供原生的 AbstractRoutingDataSource ,參考一些文檔自己實現切換。

1. Mybatis-Plus的多數據源的優缺點

優勢:
數據源分組,適用於多種場景 純粹多庫 讀寫分離 一主多從 混合模式。
簡單集成Druid數據源監控多數據源,簡單集成Mybatis-Plus簡化單表,簡單集成P6sy格式化sql,簡單集成Jndi數據源。
簡化Druid和HikariCp配置,提供全局參數配置。
提供自定義數據源來源(默認使用yml或properties配置)。
項目啓動後能動態增減數據源。
使用spel動態參數解析數據源,如從session,header和參數中獲取數據源。(多租戶架構神器)
多層數據源嵌套切換。(一個業務ServiceA調用ServiceB,ServiceB調用ServiceC,每個Service都是不同的數據源)
使用正則匹配或spel表達式來切換數據源(實驗性功能)。

劣勢:
不能使用多數據源事務(同一個數據源下能使用事務),網上其他方案也都不能提供。
如果你需要使用到分佈式事務,那麼你的架構應該到了微服務化的時候了。
如果呼聲強烈,項目達到800 star,作者考慮集成分佈式事務。
PS: 如果您只是幾個數據庫但是有強烈的需求分佈式事務,建議還是使用傳統方式自己構建多套環境集成atomic這類,網上百度很多。
建議:
本框架只做 切換數據源 這件核心的事情,並不限制你的具體操作,切換了數據源可以做任何CRUD。
配置文件所有以下劃線 _ 分割的數據源 首部 即爲組的名稱,相同組名稱的數據源會放在一個組下。
切換數據源即可是組名,也可是具體數據源名稱,切換時默認採用負載均衡機制切換。
默認的數據源名稱爲 master ,你可以通過spring.datasource.dynamic.primary修改。
方法上的註解優先於類上註解。
強烈建議在 主從模式 下遵循普遍的規則,以便他人能更輕易理解你的代碼。
主數據庫 建議 只執行 INSERT UPDATE DELETE 操作。
從數據庫 建議 只執行 SELECT 操作

2. 實現步驟

2.1 添加pom配置

<dependency>
  <groupId>com.baomidou</groupId>
  <artifactId>dynamic-datasource-spring-boot-starter</artifactId>
  <version>${version}</version>
</dependency>

2.2 配置數據源

spring:
  datasource:
    dynamic:
      primary: master #設置默認的數據源或者數據源組,默認值即爲master
      datasource:
        master:
          username: root
          password: 123456
          driver-class-name: com.mysql.jdbc.Driver
          url: jdbc:mysql://xx.xx.xx.xx:3306/dynamic
        slave_1:
          username: root
          password: 123456
          driver-class-name: com.mysql.jdbc.Driver
          url: jdbc:mysql://xx.xx.xx.xx:3307/dynamic
        slave_2:
          username: root
          password: 123456
          driver-class-name: com.mysql.jdbc.Driver
          url: jdbc:mysql://xx.xx.xx.xx:3308/dynamic

以上會配置一個默認庫master,一個組slave下有兩個子庫slave_1,slave_2

3. 切換數據源

使用 @DS 切換數據源
@DS 可以註解在方法上和類上,同時存在方法註解優先於類上註解。
註解在service實現或mapper接口方法上,但強烈不建議同時在service和mapper註解。 (可能會有問題)
註解 結果
沒有@DS 默認數據源
@DS(“dsName”) dsName可以爲組名也可以爲具體某個庫的名稱
例:

@Service
@DS("slave")
public class UserServiceImpl implements UserService {

  @Autowired
  private JdbcTemplate jdbcTemplate;

  public List<Map<String, Object>> selectAll() {
    return  jdbcTemplate.queryForList("select * from user");
  }
  
  @Override
  @DS("slave_1")
  public List<Map<String, Object>> selectByCondition() {
    return  jdbcTemplate.queryForList("select * from user where age >10");
  }
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章