每個微服務的資料自主性
提示
本內容節錄自《容器化 .NET 應用程式的 .NET 微服務架構》(.NET Microservices Architecture for Containerized .NET Applications) 電子書,可以在 .NET Docs 上取得,或免費下載可供離線閱讀的 PDF。
微服務架構的重要規則是,每個微服務必須擁有自己的網域資料和邏輯。 就像完整的應用程式擁有其邏輯和資料,每個微服務也必須在自發的生命週期裡擁有其邏輯和資料,並且每個微服務有獨立的部署。
這表示,網域的概念模型將在各個子系統或微服務之間有所不同。 請考慮企業應用程式,其中客戶關係管理 (CRM) 應用程式、交易式購買子系統和客戶支援子系統每個都呼叫不同的客戶實體屬性和資料,而每一者都使用不同的繫結內容 (BC)。
這個原則非常類似網域導向設計 (DDD),其中每個繫結內容或自發子系統或服務必須擁有其網域模型 (資料加上邏輯和行為)。 每個 DDD 繫結內容相互關聯到一個商務微服務 (一或多個服務)。 我們會在下一節從這點繼續展開繫結內容模式。
在另一方面,許多應用程式中使用的傳統 (單一資料) 方法,是單一的集中式資料庫或幾個資料庫。 這通常是標準化的 SQL 資料庫,用於整個應用程式及其所有內部子系統,如圖 4-7 所示。
圖 4-7. 資料自主性比較:單一資料庫與微服務
在傳統的方法中,所有服務之間會共用單一資料庫,通常使用分層架構。 在微服務方法中,每個微服務都擁有其模型/資料。 集中式資料庫方法一開始看起來較簡單,而且似乎能重複使用不同子系統中的實體,讓一切一致。 但事實上,您會得到很大的資料表,它們會服務許多不同子系統,且包含在大部分情況下不需要的屬性和資料行。 就像是嘗試使用相同的實體地圖來進行短距離的健走、一整天的汽車旅行,以及學習地理。
具有通常單一關聯式資料庫的單一應用程式有兩個重要優點:ACID 交易和 SQL 語言,兩者都可在與您應用程式相關的的所有資料表和資料之間使用。 這種方法可用來輕鬆地撰寫查詢,以結合來自多個資料表的資料。
不過,當您移到微服務架構時,資料存取變得複雜許多。 即使是在微服務或繫結內容中使用 ACID 交易時,也請務必考慮每個微服務所擁有的資料是該微服務的私密資料,而且只應該透過其 API 端點 (REST、gRPC、SOAP 等) 進行同步存取或透過傳訊 (AMQP 或類似工具) 進行非同步存取。
封裝資料可確保微服務鬆散偶合,而且彼此獨立地持續改進。 如果多個服務存取相同的資料,結構描述更新便需要所有服的協調性更新。 這會破壞微服務生命週期自主。 但分散式資料結構表示您無法讓單一 ACID 交易跨越微服務。 這又表示當商務程序跨越多個微服務時,您必須使用最終一致性。 這要實作時比簡單的 SQL 聯結難上許多,因為您無法建立完整性條件約束,或在不同的資料庫之間使用分散式交易,我們將在稍後說明。 同樣地,許多其他關聯式資料庫功能無法跨多個微服務使用。
更進一步地說,不同的微服務通常使用不同「種類」的資料庫。 現代應用程式會存放和處理多種種類的資料,且關聯式資料庫不一定是最佳選擇。 對於某些使用案例,例如 Azure CosmosDB 或 MongoDB 的 NoSQL 資料庫,可能會有更方便的資料模型,並提供較 SQL Server 或 Azure SQL Database 等 SQL 資料庫更佳的效能和延展性。 在其他情況下,關聯式資料庫仍是最好的方法。 因此,以微服務為基礎的應用程式通常會使用 SQL 和 NoSQL 資料庫的混合,這有時稱為混合持續性 (Polyglot Persistence) 方法。
資料儲存體的資料分割、混合持續架構有許多優點。 這些包括鬆散偶合的服務和更佳的效能、延展性、成本與管理能力。 不過,它可能導致某些分散式資料管理的挑戰,如本章稍後的識別領域模型界限中所說明。
微服務和繫結內容模式之間的關聯性
微服務概念衍生自網域導向設計 (DDD) 中的繫結內容 (BC) 模式。 DDD 透過將大型模型分成多個 BC 並且明確指定其界限來處理大型模型。 每個 BC 必須有自己的模型和資料庫。同樣地,每個微服務也擁有其相關資料。 此外,每個 BC 通常都有它自己的通用語言,協助軟體開發人員和網域專家之間的溝通。
通用語言中的詞彙 (主要為領域實體) 在不同繫結內容中可以有不同名稱,即使是不同領域實體都共用相同的識別 (亦即,用來從儲存體讀取實體的唯一識別碼)。 比方說,在使用者設定檔繫結內容中,使用者網域實體可能會與排序之繫結內容中的買方網域實體共用身分識別。
因此,微服務就像是繫結內容,但它也會指定其為一種分散式服務。 它針對每個繫結內容建置為個別的處理序,且必須使用前文所述的分散式通訊協定,例如 HTTP/HTTPS、WebSockets 或 AMQP。 不過,繫結內容模式並未指定繫結內容是否為分散式服務,還是它只是單一部署應用程式中的邏輯界限 (例如泛型子系統)。
值得強調的一點是,為每個繫結內容定義服務是個不錯的起點。 但您不必將設計限制於此。 有時您必須設計由數個實體服務組成的繫結內容或商務微服務。 但最後,這兩種模式 - 繫結內容和微服務 - 密切相關。
DDD 藉由以分散式微服務的形式得到真實界限而從微服務獲益。 但如同不要在微服務之間共用模型,您可能也不想在繫結內容中共用模型。
其他資源
Chris Richardson. 模式:每個服務的資料庫
https://microservices.io/patterns/data/database-per-service.htmlMartin Fowler。 BoundedContext
https://martinfowler.com/bliki/BoundedContext.htmlMartin Fowler。 PolyglotPersistence
https://martinfowler.com/bliki/PolyglotPersistence.htmlAlberto Brandolini. 含內容對應的策略性領域導向設計
https://www.infoq.com/articles/ddd-contextmapping