<<ABP文檔 - 框架>> 1.5 多租戶

来源:http://www.cnblogs.com/kid1412/archive/2016/10/19/5978314.html
-Advertisement-
Play Games

文檔目錄 本節內容: 什麼是多租戶 多部署 - 多資料庫 單部署 - 多資料庫 單部署 - 單資料庫 單部署 - 混資料庫 多部署 - 單/多/混 資料庫 ABP中的多租戶 啟用多租戶 宿主與租戶 會話 數據過濾 IMustHaveTenant 介面 IMayHaveTenant 介面 補充提醒 在 ...


文檔目錄

 

本節內容:

 

什麼是多租戶

維基百科:“軟體多租戶是一個軟體架構,軟體只有一個實例運行在伺服器,並服務於多個租戶。一個租戶包含一組用戶,他們擁有指定許可權,共同訪問一個軟體實例。一個多租戶架構,應用程式為每個租戶提供一個專屬於他們的數據、配置、用戶管理、租戶特有的功能和屬性。多租戶架構而多實例框架抽象而成,多實例架構是把每個實例看成一個租戶。“

多租戶通常用來創建Saas(軟體作為服務)應用(雲計算)。多租戶有多種架構:

 

多部署 - 多資料庫

這種實際上不算多租戶,但是,如果我們為每個客戶(租戶)運行應用的一個實例,並使用一個獨立的資料庫,那麼我們就可以在一臺伺服器上為多個租戶服務,我們只要確保應用的多個實例不要在一個伺服器的環境下互相衝突就行。

為一個不是為多租戶設計,但已經在運行的應用,提供了可能性。這種方式雖然使得創建一個不考慮多租戶的應用相對容易,但在安裝、使用和維護方面有些問題。

 

單部署 - 多資料庫

用這種方式,我們在一個伺服器上運行應用的單一實例,我們有一個主(宿主)資料庫存儲租戶元數據(像租戶名和子域),併為每個租戶維護一個隔離的資料庫。我們一旦識別當前租戶(例如:從子域或從一個用戶登錄窗體),就切換到該租戶的資料庫里執行操作。

用這種方式,我們應該在設計應用時,在某些層面上設計成多租戶,但應用的大部分還是不依賴於多租戶。

我們應該為每個租戶創建並維護一個隔離的資料庫,包括數據遷移。如果我們有多租戶就需要維護它們專有的資料庫,在應用更新時,可能就需要花很長的時間進行資料庫結構遷移。由於我們有租戶的隔離的資料庫,所以我們可以單獨地備份各自的資料庫,同樣在租戶要求下,我們也可以移動租戶資料庫到一個更強大的伺服器。

 

單部署 - 單資料庫

這是最純粹的多租戶架構:我們只在一臺伺服器上部署應用的單個實例和單個資料庫。我們在每個表(關係型資料庫)里用一個TenantId(租戶Id或類似的)欄位來區分隔離每個租戶數據。

這種方式易於安裝和維護,但難於創建這種應用,因為我們必須防止一個租戶讀或寫其它租戶數據。我們得為每次的資料庫讀取(select)操作添加TenantId來過濾。同樣,我們也在每次寫資料庫時進行檢查當前實體是否與當前租戶相關,這就是乏味和易犯錯的地方。但ABP自動使用數據過濾技術幫我們解決這些問題。

這種方式在有很多租戶和大數據量的情況下,可能帶來性能問題,我們可以使用表分區或其它資料庫特性來解決這個問題。

 

單部署 - 混資料庫

我們可能想存儲租戶數據到一個資料庫,但想為有需要的租戶創建單獨的資料庫。例如,我們可以存儲租戶的大數據到各自的資料庫,但其它的都保存到另一資料庫。

 

多部署 - 單/多/混 資料庫

最後,我們可能想部署我們的應用到多個伺服器(像分散式伺服器集群)獲得更好地性能、實用性和擴展性。這是一種依賴於資料庫的方式。

 

ABP中的多租戶

ABP可用於上述所描述的場景。

 

啟用多租戶

預設情況多租戶是禁用的,我們可以在我們模塊的PreInitialize(預初始化)里啟用它,如下:

Configuration.MultiTenancy.IsEnabled = true; 

 

宿主與租戶

首先我們要在多租戶系統里定義兩個術語:

  • Tenant(租戶):一個客戶,擁有多個用戶、角色、許可、設置等,並要單獨地使用這個應用。一個多租戶應用可能有多個租戶,每個租戶有它自己的帳戶、聯繫人、產品及其它。所以當我們說一個“Tenant user(租戶用戶)”,表示一個租戶下的一個用戶。
  • Host(宿主):宿主是單例的(就一個宿主),這個宿主負責創建和管理租戶,所以“Host user(宿主用戶)”擁有更高級別,不依賴於租戶,並能控制租戶。

 

會話(Session)

ABP定義了IAbpSession介面,用來獲取user(用戶)和tenant ids(租戶Id)。該介面在多租戶系統中預設情況下獲取當前租戶Id,因此它能基於租戶Id過濾數據。有如下規則:

  • 如果用戶Id和租戶Id都為null,當前用戶尚未登錄到系統,所以我們不知道它是宿主用戶還是租戶用戶。這種情況下,用戶不能訪問需要授權的內容。
  • 如果用戶Id不為null,租戶Id為null,我們就可以知道當前用戶為宿主用戶。
  • 如果用戶id不為null,租戶Id也不為null,我們就可以知道當前用戶為租戶用戶。

查看會話文檔獲取更多相關信息。

 

數據過濾

在多租戶單資料庫方式里,我們必須添加一個TenantId(租戶Id)過濾,從資料庫中只獲取當前租戶的實體。當你的實體實現IMustHaveTenant和ImayHaveTenant兩個介面中的一個,ABP就會自動做到這點。

 

IMustHaveTenant 介面

該介面通過定義TenantId屬性為不同租戶區分實體。如下所示,一個實體實現IMustHaveTenant:

public class Product : Entity, IMustHaveTenant
{
    public int TenantId { get; set; }

    public string Name { get; set; }

    //...other properties
}

因此ABP知道這是一個特定租戶的實體並自動與其它租戶的實體分離。

 

IMayHaveTenant 介面

我們有時需要在宿主與租戶之間共用一個實體,所以一個實體可能是宿主的或租戶的。IMayHaveTenant介面同樣定義了TenantId屬性(類似於IMustHaveTenant),但它是nullable(可空的)。如下所示,一個實體實現IMayHaveTenant:

public class Role : Entity, IMayHaveTenant
{
    public int? TenantId { get; set; }

    public string RoleName { get; set; }

    //...other properties
}

我們可以使用兩樣的role類來存儲宿主角色和租戶角色,在這種情況下,靠TenantId屬性來區分是宿主實體還是租戶實體。如果為null表示這是一個宿主實體,否則它就是一個租戶實體,它的值就是租戶Id。

 

補充提醒:

IMayHaveTenant沒有IMustHaveTenant那麼通用。例如:一個Product(產品)類不能是IMayHaveTenant,因為它跟應用功能切實相關的,而與租戶的管理無關。所以在使用IMayHaveTenant介面時要格外小心,畢竟維護共用於宿主與租戶的代碼比較難。

當你定義一個實體類型為IMustHaveTenant或IMayHaveTenant後,在創建一個新實體時應該特意支設置TenantId(儘管ABP會嘗試把當前TenantId賦給它,但某些情況下不會成功,尤其是使用IMayHaveTenant的實體)。大部分情況,這個TenantId屬性是唯一需要處理的點,當你寫LINQ時,不需要顯式地在where條件里寫TenantId過濾,因為它會自動地被過濾。

 

在宿主與租戶間切換

在多租戶應用資料庫上,我們應該知道當前租戶,預設情況下,可以從IAbpSession中獲取(如之前所述)。但我們可以改變這種行為,切換到其它租戶的資料庫上,例如:

public class ProductService : ITransientDependency
{
    private readonly IRepository<Product> _productRepository;
    private readonly IUnitOfWorkManager _unitOfWorkManager;

    public ProductService(IRepository<Product> productRepository, IUnitOfWorkManager unitOfWorkManager)
    {
        _productRepository = productRepository;
        _unitOfWorkManager = unitOfWorkManager;
    }

    [UnitOfWork]
    public virtual List<Product> GetProducts(int tenantId)
    {
        using (_unitOfWorkManager.Current.SetTenantId(tenantId))
        {
            return _productRepository.GetAllList();
        }
    }
}

SetTenantId確保我們工作於給定的租戶的數據,獲取方式依資料庫而定:

  • 如果給定的租戶有特定的資料庫,它切換到這個資料庫,從中獲取產品。
  • 如果給定的租戶沒有特定的資料庫(例如:單資料庫方式),它自動添加TenantId過濾到查詢里,只獲取給定租戶的產品。

如果我們不使用SetTenantId,如前面所說,將從會話中獲取TenantId。這裡有些提醒和最佳實踐:

  • 使用SetTenantId(null)可切換到宿主。
  • 如果沒有特殊情況,要像示例那樣,在using塊里使用SetTenantId,因為它會在塊後面自動還原TenantId的值,並且調用GetProducts方法的代碼也會像調用前那樣工作。
  • 如果有需要,你可以塊里嵌套使用SetTenantId 
  • 由於_unitOfWorkManger.Current僅在同一工作單元內可用,所以確保你的代碼是運行在同一個工作單元內。

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 介紹 本篇文章主要介紹在oracle中如果創建自增長表,這裡要用到序列。 create table tb_student ( id NUMBER(10) not null, createtime DATE not null, constraint PK_tb_student primary key ...
  • MYSQL中可以通過內外鍵鏈接,將有關係的表中數據合併到一起進行條件篩選: 首先創建兩個新表,數據如下: student 表數據: score 表數據: 可以看到students表中stu_id為16048008的記錄對應score表沒有數據; 1.當進行內連接時,系統會自動忽略兩個表中對應不起來的 ...
  • -- 1.給下麵的查詢填空(使用<#>標記),以獲得其後的結果。 SELECT e.emp_id, e.fname,e.lname,b.`name` FROM employee e INNER JOIN <1> b ON e.assigned_branch_id = b.<2>; 結果如下: --2 ...
  • Oracle唯一索引、普通索引及約束的關係 在總結索引掃描類型前(不同於前面總結的五大類索引類型,索引類型主要是索引類別的劃分,而索引掃描類型是索引在進行索引掃描時的具體方法),需要瞭解唯一索引、非唯一索引(普通索引)以及約束的關係。這是因為對於索引掃描類型的具體探討上,需要根據“唯一索引”、“非唯 ...
  • OS X中的操作 一、OS X如何打開活動監控器 然後直接拖到dock中 Terminal 中的操作 一、如何開啟apache 在終端輸入sudo apachectl -k start 二、查看指定的埠號被占用的情況 1、在終端中輸入sudo lsof -i tcp:port 將port換成需要查 ...
  • 在阿裡雲主機上,操作系統是Centos6,php版本是5.3。因為安裝Yii2.0的需要,我要升級php為5.4。因為還有5.5和5.6,當然要升到高版本了。我決定升到5.6。 首先,按照這裡的步驟 http://zhidao.baidu.com/link?url=8w_-5AhIIWNfbenvC ...
  • win10或win7 C盤複製文件等遇到"錯誤0x80070522:客戶端沒有所需的特權" 在運行中輸入 icacls c:\ /setintegritylevel M ...
  • 距離ubuntu最新版發佈已經差不多半年了,博主近來對linux系統有了興趣,奈何資金不足無法購置一臺新機來安裝ubuntu。所以想到了虛擬機。 虛擬機的選擇 VMwareWorkstation,功能強大,虛擬機的顯卡也不錯。VMware缺點是很不綠色,會對你的系統有一些影響。而vmare work ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...