精品欧美一区二区三区在线观看 _久久久久国色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技術棧公眾號

日韩免费观看av| 亚洲第一在线视频| 亚洲图片在线观看| 国产绿帽一区二区三区| 国产综合欧美| 日韩精品在线播放| 黄色片视频在线| 手机av免费在线| 91原创在线视频| 国产精品自产拍在线观| 国产一级av毛片| 欧美日韩激情| 精品国产免费一区二区三区香蕉| 男女视频网站在线观看| 1769视频在线播放免费观看| 国产精品99久久久久久久vr | 高清中文字幕mv的电影| 日本韩国欧美| 一区二区高清免费观看影视大全| 欧美大香线蕉线伊人久久| 亚洲专区在线播放| 国产精品女主播一区二区三区| 色综合伊人色综合网站| 亚洲国产精品自拍视频| 9999精品视频| 在线亚洲高清视频| 一级黄色片播放| 日韩a在线观看| 国产成人av一区二区三区在线| 国产精品观看在线亚洲人成网| 国产一级做a爰片在线看免费| 菠萝蜜一区二区| 亚洲国产三级网| 91亚洲精品久久久蜜桃借种| 一区一区三区| 亚洲成人午夜电影| 日本精品免费视频| 97视频在线观看网站| 久久综合九色综合97_久久久| 97视频资源在线观看| 中文字幕av无码一区二区三区| 在线一区视频| 久久免费少妇高潮久久精品99| 美国黄色片视频| 日韩理论片av| 伊人久久久久久久久久久久久| 黄色污在线观看| aaa国产精品| 欧美一区二区美女| 日韩欧美理论片| 欧美啪啪网站| 欧美日韩一区二区三区视频| 爱情岛论坛成人| 免费电影日韩网站| 狠狠色狠色综合曰曰| 男人日女人逼逼| 黄毛片在线观看| 五月婷婷激情综合| 免费日韩在线观看| 菠萝蜜视频国产在线播放| 亚洲欧洲日韩在线| 精品久久免费观看| 黄色在线免费网站| 亚洲欧美电影院| 粉嫩av一区二区三区天美传媒 | 久久99精品久久久久久青青91| 国产女人18水真多毛片18精品| 欧美.www| 高清欧美性猛交xxxx黑人猛交| 国产精品99无码一区二区| 亚洲小说欧美另类社区| 7777精品久久久久久| 亚洲影院在线播放| 日韩va欧美va亚洲va久久| 国产精品香蕉av| 国产在成人精品线拍偷自揄拍| 激情五月婷婷综合网| 97se视频在线观看| 农村少妇久久久久久久| 91亚洲男人天堂| 日韩精品最新在线观看| 免费日本一区二区三区视频| 亚洲码国产岛国毛片在线| 亚洲精品无码国产| 中文在线8资源库| 欧美在线一二三| 一级黄色大片儿| 成人春色在线观看免费网站| 亚洲欧美福利视频| 啪啪一区二区三区| 在线欧美一区| 国产精品成人av性教育| 国产女人18毛片18精品| 成人看片黄a免费看在线| 欧美日本韩国一区二区三区| 五月婷婷在线视频| 亚洲成av人片| 向日葵污视频在线观看| 亚洲网一区二区三区| 亚洲免费中文字幕| 久久国产波多野结衣| 国产情侣久久| 国产原创欧美精品| 性感美女福利视频| 亚洲人成伊人成综合网小说| 18禁免费观看网站| 色999韩欧美国产综合俺来也| 亚洲第一精品夜夜躁人人爽 | 亚洲色图在线视频| 无码人妻精品一区二区三区在线| 国产精品美女午夜爽爽| 亚洲精品国产品国语在线| 色偷偷男人天堂| 99精品国产99久久久久久福利| 国产美女久久久| 视频在线观看你懂的| 亚洲色图欧美偷拍| 成年人视频在线免费| 91精品啪在线观看国产爱臀| 国产一区二区三区高清在线观看| 久久免费播放视频| 麻豆极品一区二区三区| 精品无人区一区二区三区 | 久久99视频精品| 特级西西444www高清大视频| 不卡的av在线播放| 日日噜噜夜夜狠狠久久丁香五月| 奇米777日韩| 亚洲精品电影在线| 久久黄色免费视频| 精品无人码麻豆乱码1区2区| 欧美日韩在线精品一区二区三区| 青春草视频在线观看| 3d成人h动漫网站入口| 国产精品免费无码| 销魂美女一区二区三区视频在线| 俄罗斯精品一区二区| 黄网站在线免费看| 欧美日韩亚洲国产综合| 欧美人妻一区二区三区| 久久久久久一区二区| 国产中文一区二区| 91破解版在线观看| 日韩亚洲欧美中文三级| 亚洲精品久久久久久国| 美腿丝袜亚洲一区| 五月天色一区| 91精品美女| 亚洲天堂色网站| 神马久久久久久久| 久久久久久久久久久黄色| 免费在线观看亚洲视频| 欧美三级自拍| 17婷婷久久www| 亚洲aⅴ乱码精品成人区| 午夜国产不卡在线观看视频| 午夜剧场免费看| 夜夜嗨一区二区| 国产一级精品aaaaa看| 黑森林国产精品av| 亚洲人成五月天| 黄色污污视频软件| 国产精品入口麻豆九色| 日本超碰在线观看| 在线观看国产精品入口| 成人精品水蜜桃| 成入视频在线观看| 精品视频久久久久久| 日韩黄色一级视频| 国产精品久久久久久一区二区三区| 亚洲这里只有精品| 欧美一区二区三区久久精品茉莉花| 91免费版黄色| 欧美男人天堂| 亚洲一区999| 国产又粗又猛视频免费| 亚洲免费看黄网站| 人妻 日韩 欧美 综合 制服| 久久成人一区| 在线免费一区| 国内自拍欧美| 国产精品av免费在线观看| 蜜桃视频网站在线观看| 欧美mv和日韩mv国产网站| 国产性xxxx高清| 国产欧美一区二区三区网站| 超碰中文字幕在线观看| 国产精品美女久久久浪潮软件| 色一情一区二区三区四区| www.久久爱.com| 45www国产精品网站| 成人免费视频| 精品播放一区二区| 懂色av中文字幕| 亚洲精品菠萝久久久久久久| 亚洲av片不卡无码久久| 久久国产日韩欧美精品| 蜜桃传媒一区二区三区| 欧美丰满日韩| 久久久久久久久四区三区| 99热播精品免费| 国内精久久久久久久久久人| 午夜在线视频播放| 亚洲国产成人av在线| 在线观看亚洲一区二区| 婷婷综合在线观看| 亚洲怡红院在线观看| 久久欧美一区二区| 男生和女生一起差差差视频| 久久综合九色| 和岳每晚弄的高潮嗷嗷叫视频| 欧美综合另类| 精品国产乱码久久久久久丨区2区 精品国产乱码久久久久久蜜柚 | 精品人妻大屁股白浆无码| 精品国产欧美日韩| 国产高清精品一区二区| 成人午夜一级| 欧美在线视频网站| 国产探花在线观看| 日韩一区二区欧美| 精品影院一区| 日韩av网址在线观看| 精品人妻aV中文字幕乱码色欲| 在线观看一区不卡| 欧美性猛交bbbbb精品| 亚洲一区二区在线免费观看视频| 992在线观看| 国产欧美一区二区精品性色 | 欧美老妇交乱视频| 在线国产情侣| 国产亚洲欧洲高清| 人成在线免费视频| 亚洲精品成人久久电影| 高h调教冰块play男男双性文| 欧美日韩精品免费观看视频| 亚洲成熟少妇视频在线观看| 精品久久久中文| 国产精品6666| 亚洲小说欧美激情另类| 天天干中文字幕| 亚洲男人的天堂在线观看| 日韩欧美视频免费观看| 欧美国产日产图区| 制服 丝袜 综合 日韩 欧美| 91麻豆福利精品推荐| 欧美大喷水吹潮合集在线观看| 特黄aaaaaaaaa真人毛片| 91caoporm在线视频| 亚洲国产精品va| 人妻一区二区三区| 精品日韩一区二区三区| 亚洲av无码乱码国产麻豆 | 萌白酱国产一区二区| 91网在线播放| 最好看的2019年中文视频| 91视频在线观看| 日韩在线观看视频免费| 伊人免费在线| 久久网福利资源网站| caopen在线视频| 欧美日韩aaaa| wwwww亚洲| 97在线观看免费高清| 天堂电影一区| 国产精品久久久久久久7电影| a∨色狠狠一区二区三区| 国产精品自拍视频| 精品一区二区三区亚洲| 国产 高清 精品 在线 a| 欧美精品国产白浆久久久久| 日本不卡二区高清三区| 日韩综合精品| 久久男人资源站| 一区二区激情| 男女无套免费视频网站动漫| 狠狠v欧美v日韩v亚洲ⅴ| 337p日本欧洲亚洲大胆张筱雨| 成人精品在线视频观看| 国产精品1000部啪视频| 国产精品情趣视频| 欧美日韩精品在线观看视频 | 波多野结衣av一区二区全免费观看| 激情久久久久久| 免费裸体美女网站| 国产一区在线观看视频| 亚洲精品乱码久久久久久蜜桃欧美| 99精品视频在线播放观看| 国产综合精品久久久久成人av| 亚洲图片你懂的| 日韩经典在线观看| 欧美亚洲禁片免费| 亚洲AV无码精品国产| 亚洲欧洲国产一区| 超碰电影在线播放| 青青久久aⅴ北条麻妃| 伊人久久大香伊蕉在人线观看热v 伊人久久大香线蕉综合影院首页 伊人久久大香 | 亚洲一区影音先锋| 亚洲欧美偷拍一区| 91精品一区二区三区久久久久久| 免费观看黄一级视频| 在线日韩日本国产亚洲| 另类视频在线| 国产精品久久久久久久av大片| 97品白浆高清久久久久久| 亚洲 日韩 国产第一区| 亚洲福利国产| 中文字幕1234区| 久久久亚洲精品一区二区三区| 青草影院在线观看| 欧洲人成人精品| 免费观看a视频| 久久精品国产久精国产一老狼| 涩涩视频在线| 99久久精品免费看国产四区| 久久精品高清| 日韩视频在线免费看| 不卡一区二区在线| 精品人妻伦九区久久aaa片| 一本久久精品一区二区| 亚洲a视频在线| 啊v视频在线一区二区三区 | lutube成人福利在线观看| 久久久午夜视频| 精品国产不卡一区二区| 日韩高清专区| 久久精品盗摄| 欲求不满的岳中文字幕| 亚洲最大成人综合| 国产农村妇女毛片精品久久| www.久久久久久.com| 成人软件在线观看| 麻豆成人av| 国产一区导航| 800av在线播放| 亚洲国产cao| 狠狠躁日日躁夜夜躁av| 欧美成人午夜视频| 青草综合视频| 一区二区三区在线观看www| 老司机免费视频久久| 亚洲一级中文字幕| 日韩欧美在线第一页| 五月婷婷在线观看视频| 性日韩欧美在线视频| 超碰精品在线| 国产曰肥老太婆无遮挡| 风间由美性色一区二区三区 | 欧美激情五月| www.欧美com| 亚洲综合色视频| 亚洲精品久久久蜜桃动漫| 欧美精品一二区| 亚洲成人影音| 欧美精品一区二区三区三州| 成人爱爱电影网址| 日韩av大片在线观看| 亚洲国模精品私拍| 中文字幕在线中文字幕在线中三区| 久久久久se| 日韩高清一区二区| 在线观看免费黄色网址| 欧美日韩精品一区视频| 四虎久久免费| 91久久国产综合久久蜜月精品| 午夜精品偷拍| 国产xxxxxxxxx| 日韩欧美极品在线观看| 国产二区在线播放| 国产欧美在线视频| 中文字幕一区二区精品区| 99国产精品免费视频| 性久久久久久久久久久久| 青青草手机在线| 国产欧美精品在线| 欧美a级片一区| 成人性生活免费看| 在线精品观看国产| 国产剧情在线| 国产区一区二区| 日韩av午夜在线观看| 三上悠亚在线观看视频| 精品蜜桃在线看| 欧美极品影院| 韩国黄色一级大片| 成人app下载| 羞羞色院91蜜桃| 欧美激情一区二区三区高清视频 | 欧美亚洲另类在线一区二区三区| 热久久久久久久| 青青草原在线免费观看视频| 日韩精品在线播放| 9999精品视频| 女人喷潮完整视频| 1区2区3区精品视频| 日韩一区二区三区不卡| 国产精品视频免费在线| 欧美视频官网| 美国美女黄色片| 亚洲国产欧美精品| 国产精品极品美女在线观看|