精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

SpringBoot多租戶三種架構實現方案詳解

開發 前端
每個租戶的數據都保存在一個物理上獨立的數據庫實例中。JDBC連接將專門指向每個數據庫,因此任何池都將按租戶進行。這里,一種通用的應用程序方法是為每個租戶定義JDBC連接池,并根據與當前登錄用戶相關聯的租戶標識符來選擇要使用的池。

環境:SpringBoot3.3.0

1. 簡介

多租戶表示應用程序的單個運行實例同時為多個客戶機(租戶)服務的體系結構。這在SaaS解決方案中非常常見。在這些系統中,隔離與各種租戶相關的信息(數據、定制等)是一個特殊的挑戰。這包括存儲在數據庫中的每個租戶擁有的數據。以下是三種常用的多租戶架構實現方案:

1.1 獨立數據庫(Separate database)

圖片圖片

每個租戶的數據都保存在一個物理上獨立的數據庫實例中。JDBC連接將專門指向每個數據庫,因此任何池都將按租戶進行。這里,一種通用的應用程序方法是為每個租戶定義JDBC連接池,并根據與當前登錄用戶相關聯的租戶標識符來選擇要使用的池。

優點:

  • 數據隔離級別高,安全性好
  • 可以根據租戶的需求進行數據庫優化和擴展
  • 備份和恢復操作相對簡單

缺點:

  • 成本較高,需要為每個租戶購買和維護獨立的數據庫實例
  • 可能存在硬件資源浪費,因為每個租戶可能只使用了數據庫的一部分功能

1.2 獨立Schema(Separate schema)

圖片

每個租戶的數據都保存在單個數據庫實例上的不同數據庫Schema中。這里有兩種不同的定義JDBC連接的方法:

  • 連接可以特定地指向每個Schema,就像單獨的數據庫方法中那樣。這是一個選項,前提是驅動程序支持在連接URL中命名默認Schema,或者池機制支持命名用于其連接的Schema。使用這種方法,我們將為每個租戶創建一個不同的JDBC連接池,使用的連接池將基于與當前登錄用戶相關聯的“租戶標識符”進行選擇。
  • 連接可以指向數據庫本身(使用某些默認Schema),但使用SQL SET schema(或類似的)命令可以更改連接。使用這種方法,我們將有一個JDBC連接池用于為所有租戶提供服務,但在使用連接之前,它將被更改為引用由與當前登錄用戶關聯的“租戶標識符”命名的模式。

優點:

  • 降低了數據庫成本,因為多個租戶共享一個數據庫實例
  • 數據隔離級別仍然較高,因為每個租戶使用獨立的模式

缺點:

  • 模式之間可能存在資源競爭和性能瓶頸
  • 備份和恢復操作可能更加復雜,因為需要針對每個模式進行單獨操作

1.3 分區數據(Partitioned (discriminator) data)

圖片

所有數據都保存在一個數據庫Schema中。通過使用分區列對每個租戶的數據進行分區。這種方法將使用單個連接池為所有租戶提供服務。但是,在這種方法中,應用程序需要對每個SQL語句添加分區列(查詢時where條件加入分區列作為查詢條件)。

優點:

  • 成本最低,因為所有租戶都共享同一個數據庫實例和模式
  • 數據訪問和查詢效率可能較高,因為數據都在同一個表中

缺點:

  • 數據隔離級別最低,可能存在安全風險
  • 需要通過應用程序邏輯來確保數據的正確隔離和訪問控制
  • 數據備份和恢復操作可能非常復雜,因為需要考慮到所有租戶的數據

接下來我會對分區數據獨立數據庫2種架構進行詳細的介紹。獨立Schema方案其實與獨立數據庫模式挺像的,如果基于MySQL其實對應的就是不同數據庫(可以是同一個MySQL實例,通過use xxx切換數據庫),基于Oracle就是對應不同的用戶上(并非schema與用戶等同)。

2. 實戰案例

2.1 分區數據

注:請先確保你當前使用的SpringBoot版本(Spring Data JPA)整合的Hibernate版本至少是6.0版本以上。

實體定義

@Entity
@Table(name = "t_person")
public class Person {
  @Id
  @GeneratedValue(strategy = GenerationType.IDENTITY)
  private Long id ;
  private String name ;
  private Integer age ;
  @TenantId
  private String tenantId ;
}

這里通過@TenantId注解標注,該字段專門用來分區租戶的,Hibernate在查詢數據時會自動添加該查詢條件,如果你使用的本地SQL(自己編寫SQL),那么需要你自行添加該條件(租戶ID條件)。

編寫DAO&Service

// DAO
public interface PersonRepository extends JpaRepository<Person, Long>, JpaSpecificationExecutor<Person> {
}
// Service
@Service
public class PersonService {


  private final PersonRepository personRepository ;
  public PersonService(PersonRepository personRepository) {
    this.personRepository = personRepository ;
  }
  // 查詢所有Person數據
  public List<Person> persons() {
    return this.personRepository.findAll() ;
  }
}

Controller接口

@GetMapping("")
public List<Person> persons() {
  return this.personService.persons() ;
}

以上是開發一個業務功能的基本操作,接下來才是重點

租戶標識解析處理

該的作用獲取當前租戶ID,這里基于ThreadLocal實現

public class TenantIdResolver implements CurrentTenantIdentifierResolver<String> {


  private static final ThreadLocal<String> CURRENT_TENANT = new ThreadLocal<>();


  public void setCurrentTenant(String currentTenant) {
    CURRENT_TENANT.set(currentTenant);
  }
  @Override
  public String resolveCurrentTenantIdentifier() {
    // 注意這里不能返回null
    return Optional.ofNullable(CURRENT_TENANT.get()).orElse("default") ;
  }


  @Override
  public boolean validateExistingCurrentSessions() {
    return true;
  }
}

上面的組件用來從當前的ThreadLocal中獲取租戶ID,接下來就是像ThreadLocal存入租戶ID。

Web攔截器

該攔截器的作用用來從請求Header中獲取租戶ID,存入ThreadLocal中。

@Component
public class TenantIdInterceptor implements HandlerInterceptor {


  private final TenantIdResolver tenantIdResolver;
  public TenantIdInterceptor(TenantIdResolver tenantIdResolver) {
    this.tenantIdResolver = tenantIdResolver;
  }


  @Override
  public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    String tenantId = request.getHeader("x-tenant-id");
    tenantIdResolver.setCurrentTenant(tenantId);
    return true ;
  }
}

最后一步就是配置hibernate,設置租戶ID的解析器。

配置租戶標識解析器

spring:
  jpa:
    properties:
      hibernate:
        '[tenant_identifier_resolver]': 'com.pack.tenant.config.TenantIdResolver'

完成以上類及配置的編寫后就實現了基于列區分(分區)的多租戶架構方案。

測試

準備數據:

圖片圖片

圖片圖片

圖片圖片

SQL執行情況:

圖片圖片

自動添加了tenant_id查詢條件。

2.2 獨立數據庫

每租戶對應一個數據庫,這需要在項目中配置多個數據源,同時提供一個數據源路由的核心類。

定義多數據源配置

你也可以將數據源的信息專門存放在數據表中。

pack:
  datasource:
    defaultDs: ds1
    config:
      ds1:
        driverClassName: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://localhost:3306/tenant-01
        username: tenant01
        password: xxxooo
        type: com.zaxxer.hikari.HikariDataSource
      ds2:
        driverClassName: com.mysql.cj.jdbc.Driver
        url: jdbc:mysql://localhost:3306/tenant-02
        username: tenant02
        password: oooxxx
        type: com.zaxxer.hikari.HikariDataSource

在Spring實現多數據源切換,可以通過繼承AbstractRoutingDataSource。

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


  private static final ThreadLocal<String> HOLDER = new InheritableThreadLocal<>() ;


  public static void set(String key) {
    HOLDER.set(key) ;
  }
  public static String get() {
    return HOLDER.get() ;
  }
  public static void clear() {
    HOLDER.remove() ; 
  }
}

配置數據源Bean

@Configuration
public class DataSourceConfig {


  @Bean
  public DataSource dataSource(MultiDataSourceProperties properties) {
    PackRoutingDataSource dataSource = new PackRoutingDataSource(properties.getDefaultDs()) ;
    Map<Object, Object> targetDataSources = new HashMap<>() ;
    // PackDataSourceProperties類僅僅就是繼承DataSourceProperties
    Map<String, PackDataSourceProperties> configs = properties.getConfig() ;
    configs.forEach((key, props) -> {
      targetDataSources.put(key, createDataSource(props, HikariDataSource.class)) ;
    });
    dataSource.setTargetDataSources(targetDataSources) ;
    return dataSource ;
  }
  private static <T> T createDataSource(PackDataSourceProperties properties, Class<? extends DataSource> type) {
    // 這里沒有考慮池的配置
    return (T) properties.initializeDataSourceBuilder().type(type).build();
  }
}

接下來定義攔截器,設置當前要操作的數據源。

Web攔截器

@Component
public class TenantIdInterceptor implements HandlerInterceptor {
  @Override
  public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    String tenantId = request.getHeader("x-tenant-id");
    DataSourceContextHolder.set(tenantId) ;
    return true ;
  }
}

以上就完成了多數據源的所有類及配置的編寫。

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2025-09-29 02:00:00

2025-02-18 16:27:01

2024-09-20 05:49:04

SpringBoot后端

2024-07-08 09:03:31

2016-09-12 14:07:14

Android 定時器

2019-11-04 08:38:45

分布式事務主流TCC

2011-09-05 12:43:23

Sencha Touc事件

2023-12-14 12:26:16

SaaS數據庫方案

2023-06-07 13:50:00

SaaS多租戶系統

2021-11-23 10:30:35

Android技術代碼

2009-12-01 09:18:22

Linux版本

2014-05-23 09:23:55

多屏

2017-07-03 18:24:39

MySQL數據冗余

2022-07-22 20:00:01

高可用路由

2022-03-22 10:24:48

Linux開源Elasticsea

2023-08-14 17:58:13

RequestHTTP請求

2024-11-06 09:28:52

架構客戶端靜態

2010-09-10 08:54:52

DIV居中

2021-11-05 21:33:28

Redis數據高并發

2021-06-24 08:52:19

單點登錄代碼前端
點贊
收藏

51CTO技術棧公眾號

97精品在线播放| 国产美女在线一区| 在线观看毛片网站| 欧美成人亚洲| 亚洲高清av在线| 黄色一级大片在线观看| 免费人成在线观看播放视频| 麻豆久久一区二区| 色综合色综合网色综合| 在线观看成人动漫| 高清av一区二区三区| 亚洲日本青草视频在线怡红院 | 这里只有精品66| 亚洲第一天堂网| 天堂成人国产精品一区| 欧美成在线视频| 中文幕无线码中文字蜜桃| 成人精品视频在线观看| 姬川优奈aav一区二区| 亚洲v国产v| 精品人妻一区二区三区含羞草 | 日本午夜在线亚洲.国产| 一区二区三区影视| 伊人成综合网伊人222| 91精品国产综合久久香蕉麻豆| 日本高清视频免费在线观看| 国产三级视频在线| 成人动漫精品一区二区| 成人精品久久久| 日日夜夜综合网| 欧美国产另类| 日韩在线资源网| 少妇光屁股影院| 国产精区一区二区| 欧美性生活一区| 免费不卡av在线| 黄色网址在线免费播放| 欧美激情中文字幕| 加勒比在线一区二区三区观看| 国产三级按摩推拿按摩| 免费人成在线不卡| 国产精品91久久| av资源免费观看| 影音国产精品| 欧美黄色片免费观看| 国产性生活大片| 欧美黄色大片在线观看| 永久免费精品影视网站| 熟女高潮一区二区三区| 人人精品亚洲| 亚洲成人网久久久| 男人女人拔萝卜视频| 国产一区 二区| 欧美美女网站色| 欧美又粗又长又爽做受| www.在线视频| 亚洲欧美日韩人成在线播放| 在线观看欧美一区| 欧美成人hd| 国产精品高潮呻吟久久| 亚洲制服中文| 国产精品ⅴa有声小说| 久久天天做天天爱综合色| 狠狠久久综合婷婷不卡| 亚洲 国产 欧美 日韩| 成人av电影在线播放| 国产精品国产精品国产专区蜜臀ah | а√天堂中文在线资源bt在线| 久久精品一二三| 亚洲成人网上| 免费黄色在线观看| 又紧又大又爽精品一区二区| avove在线观看| 天堂av在线电影| 亚洲成av人片在www色猫咪| 免费看国产曰批40分钟| 69久成人做爰电影| 91久久线看在观草草青青| 91最新在线观看| japansex久久高清精品| 精品欧美一区二区三区精品久久| a级片在线观看视频| 久久99影视| 精品国产一区二区三区四区在线观看 | 999一区二区三区| 91白丝在线| 欧美性猛交xxx| 日韩一区二区三区不卡视频| 精品99re| 亚洲韩国青草视频| 欧美熟妇激情一区二区三区| 国产国产精品| 性色av一区二区三区免费| 久久不卡免费视频| 老色鬼精品视频在线观看播放| 亚洲va码欧洲m码| 天堂av一区二区三区| 国产亚洲1区2区3区| 伊人久久大香线蕉成人综合网 | 精品国产乱码久久久久久果冻传媒| 在线观看欧美日韩国产| 久草成人在线视频| 性欧美长视频| 亚洲一区二区免费在线| 午夜视频在线播放| 国产精品久久三| 日韩av中文字幕第一页| 51一区二区三区| 欧美不卡一区二区三区| 四虎永久免费在线观看| 影音先锋日韩在线| 国产999精品久久久影片官网| 国产精品羞羞答答在线| 2023国产精品自拍| 好吊色视频988gao在线观看| 欧美电影h版| 日韩欧美一区二区不卡| 人妻精品久久久久中文| 很黄很黄激情成人| 日韩av免费看网站| 亚洲欧美另类综合| 国产精品福利电影一区二区三区四区 | jizzjizz在线观看| 黄色一区二区三区| 婷婷中文字幕在线观看| 国产一区二区三区四区二区| 欧美高清视频在线播放| 一级片视频网站| 久久久精品国产免费观看同学| av在线com| 成人污污www网站免费丝瓜| 国产亚洲免费的视频看| 91香蕉在线视频| 国产成人免费高清| 亚洲 欧洲 日韩| 国产成人精品一区二区三区视频 | 成人欧美一区二区三区在线| 国产免费永久在线观看| 精品久久香蕉国产线看观看gif| 三年中文在线观看免费大全中国| 国产免费久久| 日本乱人伦a精品| 青青青手机在线视频观看| 亚洲午夜免费视频| 特种兵之深入敌后| 亚洲一本二本| 成人黄色av网站| 欧美性天天影视| 欧美亚洲综合在线| 精品人妻一区二区三区蜜桃视频| 午夜综合激情| 久久av一区二区三区亚洲| 密臀av在线| 亚洲成avwww人| 久久久久久久久99| 成人一区二区视频| 国产欧美精品aaaaaa片| 99a精品视频在线观看| 欧美激情xxxxx| 亚洲产国偷v产偷v自拍涩爱| 一区二区免费视频| 成人做爰www看视频软件| 亚洲精品九九| 精品一区二区视频| 中文字幕在线官网| 国产一区二区日韩精品欧美精品| www五月天com| 欧美国产一区在线| 污污动漫在线观看| 香蕉视频官网在线观看日本一区二区| 成人黄色片网站| 污污片在线免费视频| 欧美变态tickle挠乳网站| 国产亚洲欧美精品久久久www | 欧美视频日韩视频在线观看| www.黄色在线| 国产在线精品一区二区夜色| 日本xxxxx18| 精品久久对白| 国产suv精品一区二区三区88区| 国产精品天堂| 欧美一区二区成人6969| 国产一级久久久| 久久日韩精品一区二区五区| 无限资源日本好片| 中文视频一区| 麻豆成人在线播放| 日韩美女在线| 欧美激情在线一区| 韩国三级在线观看久| 91精品久久久久久久99蜜桃| 国产精品1000| 国产精品视频一区二区三区不卡| 午夜视频在线观| 国产深夜精品| 一区二区国产日产| 欧美日韩夜夜| 亚洲一区二区三区777| 亚洲人体视频| 两个人的视频www国产精品| 全部免费毛片在线播放一个| 欧美午夜免费电影| 福利一区二区三区四区| 中文字幕一区二区三区蜜月| 日本丰满少妇裸体自慰 | 欧美熟妇精品一区二区蜜桃视频| 日韩精品免费视频人成| 久久久久99精品成人片| 手机在线电影一区| 免费毛片一区二区三区久久久| 亚洲老司机网| 国产精品成av人在线视午夜片| 亚洲丝袜一区| 最近2019年好看中文字幕视频| 色呦呦免费观看| 这里只有精品电影| 中文永久免费观看| 都市激情亚洲色图| 精品无码免费视频| 中文字幕综合网| 成人激情五月天| 久久网这里都是精品| 国产精品成人99一区无码| 国产乱人伦精品一区二区在线观看 | 免费大片黄在线| 精品国产乱子伦一区| 成年人视频在线免费看| 亚洲精品国产视频| 日本黄色激情视频| 久久久www成人免费毛片麻豆 | 伊人久久综合视频| 亚洲免费资源在线播放| 非洲一级黄色片| 91美女福利视频| 香蕉网在线视频| 美女视频网站黄色亚洲| 亚洲精品乱码久久久久久自慰| 亚洲人成毛片在线播放女女| 日本一区二区免费高清视频| 国产精品中文字幕亚洲欧美| 国产一区二区三区无遮挡| 国产一区二区三区视频在线| 国产日韩换脸av一区在线观看| 日韩高清在线| 992tv成人免费视频| 亚洲夜夜综合| 久久福利网址导航| 免费大片在线观看www| 中文字幕精品网| 亚洲 美腿 欧美 偷拍| 日韩av在线免费播放| 亚洲国产精品久久久久爰性色 | 四虎在线视频免费观看| 精品久久久久久久久久久久久久久久久| 一级黄在线观看| 欧美高清激情brazzers| 中文字幕免费高清网站| 欧美午夜无遮挡| 亚洲精品视频在线观看免费视频| 亚洲成人资源在线| 免费一级特黄特色大片| 一区av在线播放| 国产精品成人av久久| 亚洲一区自拍偷拍| 激情五月婷婷小说| 午夜精品免费在线| 国产又大又黑又粗免费视频| 婷婷综合在线观看| 国产在线一二区| 欧美日韩一区二区精品| 草久久免费视频| 黄色成人在线播放| 国产美女www| 91福利在线免费观看| 高潮无码精品色欲av午夜福利| 欧美日本一道本| 国产农村妇女毛片精品久久| 日韩一区二区三区电影 | 国产激情精品久久久第一区二区| 在线观看免费看片| www.亚洲色图.com| 久操视频在线观看免费| 中文字幕一区在线观看| 国产va在线播放| 欧美性猛交xxxx| 久久久999久久久| 69av一区二区三区| 亚洲精品一区二区三区区别| 亚洲精品wwww| 福利视频在线播放| 中文字幕亚洲一区在线观看| 国产网站在线免费观看| 国内伊人久久久久久网站视频| 亚洲色图美国十次| 国产999视频| 韩国三级成人在线| 精品国产综合久久| 欧美电影免费观看高清| 欧美视频在线观看视频| 免费在线亚洲欧美| 性生活在线视频| 95精品视频在线| 日本裸体美女视频| 精品久久久国产| 中日韩精品视频在线观看| 欧美日韩一区二区三区在线| 成人免费公开视频| 日韩中文字幕精品视频| 91丝袜在线| 91精品久久久久久久| 综合色就爱涩涩涩综合婷婷| 亚洲激情啪啪| 亚洲毛片播放| 熟妇无码乱子成人精品| 久久伊99综合婷婷久久伊| 亚洲欧美精品久久| 日本精品一区二区三区高清| 国产成人毛毛毛片| 亚洲片在线观看| av免费不卡国产观看| 国产男女猛烈无遮挡91| 红杏一区二区三区| 热这里只有精品| 久久免费国产| www男人天堂| 一区二区三区免费在线观看| 黄色一级视频免费看| 精品国产1区二区| 中文字幕有码在线视频| 国产成人自拍视频在线观看| 麻豆一区在线| 成年人三级视频| 日韩av中文在线观看| 波多野结衣影院| 一区二区三区在线视频观看| 免费在线观看av的网站| 亚洲热线99精品视频| 国产经典三级在线| 91精品视频一区| 国产精品伦理久久久久久| aaa毛片在线观看| 99在线热播精品免费| 日产欧产va高清| 日韩亚洲欧美一区二区三区| 成人免费黄色网页| 国产精品久久久久久久久借妻| 亚洲影院天堂中文av色| 精品国产av无码一区二区三区| 黑人精品欧美一区二区蜜桃| 国产人妻大战黑人20p| 欧洲中文字幕精品| 你懂的视频在线| 欧美亚洲另类在线| 精品日韩一区| www.日日操| 国产婷婷色一区二区三区在线| 波多野结衣一二区| 亚洲欧美精品中文字幕在线| 韩漫成人漫画| 日本一区二区三区四区高清视频 | 国产妇女馒头高清泬20p多| 国产a级毛片一区| 久久这里只有精品国产| 精品久久免费看| 女同视频在线观看| 久久久久网址| 久久婷婷久久| 欧美成人另类视频| 欧美美女直播网站| a黄色片在线观看| 国产日韩在线视频| 中文视频一区| 免费不卡的av| 色婷婷综合视频在线观看| 国产三级在线免费观看| 国产精品无av码在线观看| 欧美大片一区| 中文文字幕文字幕高清| 精品二区三区线观看| 激情综合闲人网| 成人精品视频在线| 国内精品久久久久国产盗摄免费观看完整版| 性猛交╳xxx乱大交| 一个色在线综合| 色在线免费视频| 国产精品视频精品| 国产精品www994| 短视频在线观看| 欧美三级视频在线观看| 国产嫩草在线视频| 欧美三级网色| 韩国av一区二区三区四区| 国产极品在线播放| 亚洲美女av在线播放| 亚洲我射av| 欧美,日韩,国产在线| 亚洲私人黄色宅男| 手机亚洲第一页| 国产精品一区久久| 亚洲性感美女99在线|