日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

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

來源: 責編: 時間:2024-05-29 08:58:35 224觀看
導讀環(huán)境:SpringBoot3.3.01. 簡介多租戶表示應用程序的單個運行實例同時為多個客戶機(租戶)服務的體系結構。這在SaaS解決方案中非常常見。在這些系統(tǒng)中,隔離與各種租戶相關的信息(數據、定制等)是一個特殊的挑戰(zhàn)。這包括存儲在

環(huán)境:SpringBoot3.3.0RIB28資訊網——每日最新資訊28at.com

1. 簡介

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

1.1 獨立數據庫(Separate database)

圖片圖片RIB28資訊網——每日最新資訊28at.com

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

優(yōu)點:RIB28資訊網——每日最新資訊28at.com

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

缺點:RIB28資訊網——每日最新資訊28at.com

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

1.2 獨立Schema(Separate schema)

圖片RIB28資訊網——每日最新資訊28at.com

每個租戶的數據都保存在單個數據庫實例上的不同數據庫Schema中。這里有兩種不同的定義JDBC連接的方法:RIB28資訊網——每日最新資訊28at.com

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

優(yōu)點:RIB28資訊網——每日最新資訊28at.com

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

缺點:RIB28資訊網——每日最新資訊28at.com

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

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

圖片RIB28資訊網——每日最新資訊28at.com

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

優(yōu)點:RIB28資訊網——每日最新資訊28at.com

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

缺點:
RIB28資訊網——每日最新資訊28at.com

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

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

2. 實戰(zhàn)案例

2.1 分區(qū)數據

注:請先確保你當前使用的SpringBoot版本(Spring Data JPA)整合的Hibernate版本至少是6.0版本以上。RIB28資訊網——每日最新資訊28at.com

實體定義

@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注解標注,該字段專門用來分區(qū)租戶的,Hibernate在查詢數據時會自動添加該查詢條件,如果你使用的本地SQL(自己編寫SQL),那么需要你自行添加該條件(租戶ID條件)。RIB28資訊網——每日最新資訊28at.com

編寫DAO&Service

// DAOpublic interface PersonRepository extends JpaRepository<Person, Long>, JpaSpecificationExecutor<Person> {}// Service@Servicepublic 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() ;}

以上是開發(fā)一個業(yè)務功能的基本操作,接下來才是重點RIB28資訊網——每日最新資訊28at.com

租戶標識解析處理

該的作用獲取當前租戶ID,這里基于ThreadLocal實現RIB28資訊網——每日最新資訊28at.com

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。RIB28資訊網——每日最新資訊28at.com

Web攔截器

該攔截器的作用用來從請求Header中獲取租戶ID,存入ThreadLocal中。RIB28資訊網——每日最新資訊28at.com

@Componentpublic 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的解析器。RIB28資訊網——每日最新資訊28at.com

配置租戶標識解析器

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

完成以上類及配置的編寫后就實現了基于列區(qū)分(分區(qū))的多租戶架構方案。RIB28資訊網——每日最新資訊28at.com

測試

準備數據:RIB28資訊網——每日最新資訊28at.com

圖片圖片RIB28資訊網——每日最新資訊28at.com

圖片圖片RIB28資訊網——每日最新資訊28at.com

圖片圖片RIB28資訊網——每日最新資訊28at.com

SQL執(zhí)行情況:RIB28資訊網——每日最新資訊28at.com

圖片圖片RIB28資訊網——每日最新資訊28at.com

自動添加了tenant_id查詢條件。RIB28資訊網——每日最新資訊28at.com

2.2 獨立數據庫

每租戶對應一個數據庫,這需要在項目中配置多個數據源,同時提供一個數據源路由的核心類。RIB28資訊網——每日最新資訊28at.com

定義多數據源配置

你也可以將數據源的信息專門存放在數據表中。RIB28資訊網——每日最新資訊28at.com

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。RIB28資訊網——每日最新資訊28at.com

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

@Configurationpublic 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();  }}

接下來定義攔截器,設置當前要操作的數據源。RIB28資訊網——每日最新資訊28at.com

Web攔截器

@Componentpublic 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 ;  }}

以上就完成了多數據源的所有類及配置的編寫。RIB28資訊網——每日最新資訊28at.com

本文鏈接:http://m.www897cc.com/showinfo-26-91376-0.htmlSpringBoot多租戶三種架構實現方案詳解

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯(lián)系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: 聊聊主流消息隊列的認證和鑒權!

下一篇: 京東二面:為什么Netty要創(chuàng)造FastThreadLocal?

標簽:
  • 熱門焦點
  • Find N3入網:最高支持16+1TB

    OPPO將于近期登場的Find N3折疊屏目前已經正式入網,型號為PHN110。本次Find N3在外觀方面相比前兩代有很大的變化,不再是小號的橫向折疊屏,而是跟別的廠商一樣采用了較為常見的
  • K60 Pro官方停產 第三方瞬間漲價

    雖然沒有官方宣布,但Redmi的一些高管也已經透露了,Redmi K60 Pro已經停產且不會補貨,這一切都是為了即將到來的K60 Ultra鋪路,屬于廠家的正常操作。但有意思的是該機在停產之后
  • 一加Ace2 Pro官宣:普及16G內存 引領24G

    一加官方今天繼續(xù)為本月發(fā)布的新機一加Ace2 Pro帶來預熱,公布了內存方面的信息。“淘汰 8GB ,12GB 起步,16GB 普及,24GB 引領,還有呢?#一加Ace2Pro#,2023 年 8 月,敬請期待。”同時
  • 石頭智能洗地機A10 Plus體驗:雙向自清潔治好了我的懶癌

    一、前言和介紹專為家庭請假懶人而生的石頭科技在近日又帶來了自己的全新旗艦新品,石頭智能洗地機A10 Plus。從這個產品名上就不難看出,這次石頭推出的并不是常見的掃地機器
  • 女孩租房開2小時空調用完100元電費引熱議:5級能耗惹不起 月薪過萬電費也交不起

    近日,江蘇蘇州一女孩租房當天充值了100元電費,開著空調不到2小時發(fā)現電費已用完。對于為什么這個快,房東表示,電表壞了這種情況很多,之前也遇到過,給租客換
  • 服務存儲設計模式:Cache-Aside模式

    Cache-Aside模式一種常用的緩存方式,通常是把數據從主存儲加載到KV緩存中,加速后續(xù)的訪問。在存在重復度的場景,Cache-Aside可以提升服務性能,降低底層存儲的壓力,缺點是緩存和底
  • 為什么你不應該使用Div作為可點擊元素

    按鈕是為任何網絡應用程序提供交互性的最常見方式。但我們經常傾向于使用其他HTML元素,如 div span 等作為 clickable 元素。但通過這樣做,我們錯過了許多內置瀏覽器的功能。
  • 2納米決戰(zhàn)2025

    集微網報道 從三強爭霸到四雄逐鹿,2nm的廝殺聲已然隱約傳來。無論是老牌勁旅臺積電、三星,還是誓言重回先進制程領先地位的英特爾,甚至初成立不久的新
  • 聯(lián)想的ThinkBook Plus下一版曝光,鍵盤旁邊塞個平板

    ThinkBook Plus 是聯(lián)想的一個特殊筆記本類別,它在封面放入了一塊墨水屏,也給人留下了較為深刻的印象。據有人爆料,聯(lián)想的下一款 ThinkBook Plus 可能更特殊,它
Top 日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不
亚洲美女诱惑| 国产精品三级久久久久久电影| 先锋影音网一区二区| 欧美亚洲免费在线| 久久免费少妇高潮久久精品99| 久久青青草综合| 欧美日韩国语| 国产精品稀缺呦系列在线| 国产在线精品二区| 亚洲欧洲日韩女同| 亚洲一卡久久| 久久九九精品99国产精品| 欧美高清视频在线| 欧美日韩国产经典色站一区二区三区| 欧美性猛交xxxx乱大交退制版| 国产精品一区二区三区久久久 | 国产香蕉久久精品综合网| 怡红院精品视频在线观看极品| 亚洲理论电影网| 欧美一区二区在线| 欧美国产在线视频| 国产精品家庭影院| 亚洲国产老妈| 亚洲一区国产| 农夫在线精品视频免费观看| 国产精品红桃| 亚洲国产成人在线视频| 亚洲女同同性videoxma| 免费在线视频一区| 国产精品日韩在线| 亚洲人成在线免费观看| 午夜精品区一区二区三| 欧美精品一区二区三区四区| 国产日韩欧美在线| 99国产欧美久久久精品| 久久免费午夜影院| 国产精品免费久久久久久| 亚洲欧洲一区二区三区| 久久国产黑丝| 国产精品麻豆va在线播放| 最新国产の精品合集bt伙计| 欧美在线观看你懂的| 欧美视频在线观看| 亚洲国产欧美在线| 欧美一区二区三区视频在线| 欧美日韩国产一中文字不卡| 在线观看一区二区视频| 午夜精品久久久久久久久久久久 | 欧美在线免费一级片| 欧美日韩一区二区在线播放| 亚洲第一级黄色片| 久久精品水蜜桃av综合天堂| 国产精品国产三级国产| 91久久精品视频| 久久人人爽国产| 国产精品婷婷午夜在线观看| 一本综合久久| 欧美极品影院| 亚洲黄一区二区三区| 看片网站欧美日韩| 国产一区二区日韩精品| 亚洲女人天堂av| 欧美三区美女| 一区二区久久久久久| 欧美极品色图| 亚洲精品免费一区二区三区| 蜜桃伊人久久| 亚洲电影在线播放| 浪潮色综合久久天堂| 国产亚洲一区在线播放| 欧美一区二区三区久久精品| 国产精品婷婷| 西瓜成人精品人成网站| 国产精品日日摸夜夜摸av| 亚洲先锋成人| 国产精品成av人在线视午夜片| 亚洲美女精品一区| 欧美理论电影在线观看| 亚洲精品免费看| 欧美日韩国产在线看| 99精品黄色片免费大全| 欧美日韩三级在线| 一区二区日韩免费看| 欧美日韩妖精视频| 亚洲私人影院在线观看| 国产精品露脸自拍| 欧美亚洲系列| 国产专区精品视频| 久久亚洲免费| 亚洲人成网站在线播| 欧美区视频在线观看| 亚洲视频第一页| 国产精品美女黄网| 欧美一区二区三区四区高清| 国产亚洲一区二区三区在线观看| 欧美在线关看| 亚洲大胆美女视频| 欧美久久久久中文字幕| 一区二区三区视频在线看 | 亚洲国产专区| 欧美日本三区| 亚洲已满18点击进入久久| 国产精品永久| 久久久精品动漫| 亚洲国产精品激情在线观看| 欧美黄污视频| 亚洲一区国产视频| 国产亚洲综合在线| 欧美18av| 在线亚洲免费| 国产日韩一区二区三区| 久久久久一区二区三区| 最近中文字幕日韩精品| 欧美日韩亚洲另类| 欧美一区二区视频在线观看| 亚洲高清在线视频| 欧美揉bbbbb揉bbbbb| 篠田优中文在线播放第一区| 狠狠综合久久| 欧美日韩喷水| 欧美一级免费视频| 亚洲高清123| 欧美特黄一区| 久久精品一区二区三区中文字幕| 亚洲国产你懂的| 国产精品免费一区二区三区观看| 久久久精品国产免费观看同学| 日韩视频在线免费观看| 国产伦精品一区二区三区高清| 久久在线观看视频| 一本一本久久a久久精品牛牛影视| 国产午夜久久| 欧美日韩国产另类不卡| 欧美在线视频一区| 亚洲美女福利视频网站| 亚洲激情电影中文字幕| 国产精品久久一区主播| 久久综合九色综合网站| 在线一区二区日韩| 黄网站色欧美视频| 欧美视频一区二区三区…| 久久裸体艺术| 亚洲一级二级在线| 亚洲国产精品福利| 国产精品日日摸夜夜添夜夜av| 男男成人高潮片免费网站| 亚洲欧美另类在线| 亚洲免费福利视频| 国内自拍视频一区二区三区| 欧美午夜精品伦理| 欧美成人蜜桃| 欧美中文字幕精品| 99热这里只有精品8| 激情国产一区二区| 国产精品欧美日韩| 欧美精品激情| 久久人人超碰| 亚洲欧美日韩中文视频| 亚洲精选视频在线| 伊人久久大香线蕉av超碰演员| 国产精品久久久久天堂| 免费在线欧美黄色| 久久国产免费| 午夜精品久久久99热福利| 99精品欧美| 亚洲国产激情| 激情久久影院| 国产日韩精品一区二区| 欧美日韩一区二区三区在线观看免| 久久亚洲影院| 久久国产精品免费一区| 亚洲一区高清| 一本综合久久| 日韩午夜免费| 最新热久久免费视频| 黑人中文字幕一区二区三区| 国产精品一卡| 国产精品福利在线观看网址| 欧美日本高清| 欧美国产精品劲爆| 老色鬼久久亚洲一区二区| 欧美一区二区三区视频免费| 亚洲伊人第一页| 一区二区三区欧美在线观看| 亚洲理论在线| 亚洲三级视频| 亚洲激情一区二区| 亚洲国产婷婷综合在线精品| 一色屋精品视频在线观看网站| 国产欧美日韩伦理| 国产精品一区二区三区观看 | 国产亚洲精品久久久久动| 国产精品伦子伦免费视频| 欧美午夜电影在线| 国产精品高清一区二区三区| 欧美网站在线| 国产精品爱久久久久久久| 欧美日韩在线一区二区三区| 欧美日韩喷水| 国产精品人人做人人爽| 国产精品丝袜久久久久久app| 国产精品视频精品| 国产欧美欧美| 国产一区欧美日韩|