在微服務中套用簡化 CQRS 和 DDD 模式
提示
本內容節錄自《容器化 .NET 應用程式的 .NET 微服務架構》(.NET Microservices Architecture for Containerized .NET Applications) 電子書,可以在 .NET Docs 上取得,或免費下載可供離線閱讀的 PDF。
CQRS 是將模型分隔為讀取資料和寫入資料的架構模式。 相關術語命令查詢隔離 (CQS) 最初是 Bertrand Meyer 在 Object Oriented Software Construction 一書中定義。 其基本概念是,您可以將系統的作業分成兩個明顯不同的分類:
[查詢]。 這些查詢會傳回結果,但不會變更系統的狀態,而且沒有副作用。
命令。 這些命令會變更系統的狀態。
CQS 是一個簡單的概念,其與相同物件內的方法 (查詢或命令) 相關。 每個方法可傳回狀態或變動狀態,但不會同時執行。 即使是單一存放庫模式物件也會遵守 CQS。 CQS 可視為 CQRS 的基本原則。
命令和查詢職責分離 (CQRS)是由 Greg Young 所引進,並由 Udi Dahan 等人強烈提倡。 它是以 CQS 原則為依據,但內容更詳細。 它可以視為根據命令和事件 (或選擇性地根據非同步訊息) 的模式。 在許多情況下,CQRS 與更進階的案例相關,像是針對讀取 (查詢) 和寫入 (更新) 使用不同的實體資料庫。 此外,更進階的 CQRS 系統可能會針對您的更新資料庫實作事件溯源 (Event Sourcing,ES),因此您只會在領域模型中儲存事件,而不會儲存目前的狀態資料。 不過,此方法未用於此指南中。 此指南使用最簡單的 CQRS 方法,該方法只會將查詢與命令隔離。
您可以將查詢作業分組為一個層級,並將命令分組為另一個層級,來達到 CQRS 的分離狀況。 每個層級都有自己的資料模型 (請注意我們是說模型,但不一定是不同的資料庫),並使用自己的模式和技術組合來建立。 更重要的是,這兩個層級可以在相同的階層或微服務中,如本指南中所使用的範例 (訂購微服務) 所示。 或者,它們可能會在不同的微服務或處理序上實作,以便可分別最佳化和擴充而不會彼此影響。
CQRS 表示會有兩個物件用於讀取/寫入作業,在其他內容中則有一個。 使用反正規化的讀取資料庫有其原因,您可以在更進階的 CQRS 文獻中了解。 但我們不會在此使用該方法,因為此處的目標在於提高查詢的彈性,而不在於使用 DDD 模式中的條件約束 (例如彙總) 來限制查詢。
eShopOnContainers 參考應用程式中的訂購微服務即為這種服務的範例。 此服務會實作以簡化的 CQRS 方法為基礎的微服務。 它會使用單一資料來源或資料庫,但有兩個邏輯模型加上異動領域的 DDD 模式,如圖 7-2 所示。
圖 7-2. 簡化的 CQRS 和 DDD 型微服務
邏輯 “Ordering” 微服務包含其 Ordering 資料庫,該資料庫可以但不必位於相同的 Docker 主機。 資料庫在相同的 Docker 主機中對開發有利,但對生產則不然。
應用程式層可以是 Web API 本身。 此處的重要設計觀點是微服務已遵循 CQRS 模式,將查詢、ViewModel (專為用戶端應用程式所建立的資料模型) 與命令、領域模型、異動分開。 此方法可確保查詢不會受限於來自 DDD 模式的限制和條件約束,這些限制和條件約束只對異動和更新才有意義,如稍後章節所述。
其他資源
- Greg Young: 事件來源系統中的版本設定 (免費閱讀線上電子書)
https://leanpub.com/esversioning/read