設計停靠站點
「沒有石頭就沒有拱門」 --馬可波羅
複製代碼
在本章中,咱們將對Contoso會議管理系統進行一個高層次的概述。這將幫助您理解應用程序的結構、集成點以及應用程序的各個部分之間的關係。git
在這裏,咱們借用Eric Evans在他的書《領域驅動設計 軟件核心複雜性應對之道(Addison-Wesley Professional, 2003)中描述的領域驅動設計(DDD)方法來描述這個高級結構。DDD是成功實現CQRS模式的先決條件雖然尚未達成廣泛的共識,但咱們團隊依然決定按照CQRS社區的慣例使用DDD裏衆多概念和方法,例如領域、限界上下文,和聚合。參考指南的第1章「上下文中的CQRS」更詳細地討論了DDD和CQRS模式之間的關係。github
在本章中,咱們使用了一些術語,咱們將在稍後定義它們。有關更多細節和可能的替代定義,請參見參考指南中的第1章「上下文中的CQRS」。數據庫
領域(Domain):領域是指Contoso會議管理系統的業務域(參考實現)。第一章「咱們的領域:Contoso會議管理系統」概述了這個領域。測試
限界上下文(Bounded Context):術語限界上下文來自Eric Evans的書。簡而言之,Evans引入這個概念是爲了將一個大型的、複雜的系統分解成更易於管理的部分。大型系統由多個限界上下文組成。每一個限界上下文都是本身包含領域模型的上下文,而且有它本身的通用語言(Ubiquitous Language)。您還能夠將限界上下文看作明肯定義了一致性邊界的自主業務組件。一個限界上下文一般經過引起事件(Raising Events)與另外一個限界上下文通訊。網站
Gary(CQRS專家)發言:
當你使用CQRS模式時,常用事件在限界上下文中進行通訊。集成還有其餘方法,好比在數據庫級共享數據。ui
上下文映射:根據Eric Evans的說法,您應該「描述模型之間的聯繫點,明確全部通訊須要的轉換,並突出任何共享的內容。」這個實踐產生了所謂的上下文映射,它有幾個用途,包括提供整個系統的概覽,以及幫助人們理解不一樣的限界上下文如何相互交互的細節。spa
訂單和註冊限界上下文:在訂單和註冊限界上下文中包含預訂、付款和註冊項。當註冊用戶與系統交互時,系統建立一個訂單來管理預訂、付款和註冊。訂單包含一個或多個訂單項。
預訂是在會議上臨時預訂一個或多個座位。當註冊用戶開始訂購會議上的一些座位時,系統會爲這些座位建立預訂。而後,其餘用戶沒法預訂這些座位。預訂保留15分鐘,在此期間,登陸人能夠經過支付座位的費用完成訂購過程。若是用戶沒有在15分鐘內付款,系統將刪除該預訂,其餘用戶能夠繼續預訂這些座位。設計
Carlos(領域專家)發言:
咱們討論過將系統保留預訂的時間期限設置成一個參數,這樣客戶能夠爲每次會議調整這個預留時間。若是咱們肯定須要這種級別的控制,那麼這多是咱們須要添加的一個特性。code
會議管理限界上下文:在這個限界上下文中,業務客戶能夠建立新的會議並管理它們。在業務客戶建立新會議以後,他可使用電子郵件和訪問代碼來訪問查看會議的詳細信息。當業務客戶建立會議時,系統生成訪問代碼。
業務客戶能夠指定如下關於會議的信息:cdn
此外,業務客戶能夠經過發佈或取消發佈來控制會議在公共網站上的可見性。
業務客戶還可使用會議管理網站查看訂單和參會者列表。
支付限界上下文:支付限界上下文負責管理會議管理系統和外部支付系統之間的交互。它將必要的付款信息轉發給外部系統,並接收付款被接受或拒絕的結果。它將支付的成功或失敗報告給會議管理系統。一開始,支付限界上下文將假定業務客戶在第三方支付系統中有一個賬戶(儘管不必定是商家賬戶),或者業務客戶將接受發~票支付。
注:在掘金,發~票是違禁詞!!!其餘網站均可以發佈,惟獨在掘金一直髮表不了。最後反覆查找,盡然是「發~票」這個詞。自我閹割至此,真是荒唐。
雖然這幾個限界上下文沒有進入Contoso會議管理系統的最終版本,可是作了一些工做。社區成員正在開發這些以及一些其餘功能,任何帶外發布和更新都將在「CQRS之旅」網站上公佈。若是您想對這些限界上下文或系統的任何其餘方面有所貢獻,請訪問項目「CQRS Journey」網站或經過cqrsjourney@microsoft.com讓咱們知道。
折扣限界上下文: 處理會議座位的購買管理和應用折扣。該會議與主系統三個已存在的限界上下文集成在一塊兒。
偶爾斷開鏈接的會議管理客戶端:這個限界上下文用於處理會議現場的管理,具備處理標籤打印、記錄參會者到達狀況和額外座位銷售的功能。
提交和進度管理限界上下文:用於處理使用Node.js編寫的論文提交和會議事件調度。
在這個版本中沒有實現候選清單功能,可是社區成員正在開發這個特性和其餘特性。任何帶外發布和更新都將在「CQRS之旅」網站上公佈。
圖1和後面的列表顯示上下文映射,該映射顯示構成完整系統的不一樣限界上下文之間的關係,所以它提供了系統如何組合在一塊兒的高級概述。儘管這個上下文映射看起來很是簡單,可是這些限界上下文的實現,以及更重要的是它們之間的交互,都是相對複雜的。這使咱們可以遭遇並處理CQRS模式和Event Sourcing的普遍問題,並有了一個豐富的源頭來獲取不少寶貴的經驗和教訓。
Gary(CQRS專家)發言:
關於CQRS項目的一個常見說法是:很難理解全部部分是如何組合在一塊兒的,特別是若是系統中有不少命令和事件。一般,您能夠對代碼執行一些靜態分析,以肯定事件和命令是在哪裏被處理的,可是很難查明它們是在哪裏發起的。在高層次上,上下文映射能夠幫助您理解不一樣限界上下文和相關事件之間的集成。維護有關命令和事件的最新文檔能夠提供更詳細的信息。另外,若是你有測試,使用命令做爲輸入,而後檢查事件,您能夠經過測試來了解某個特定命令的預期結果(參見第4章「擴展和提升訂單和註冊限界上下文」中的測試示例)。
圖1顯示了構成Contoso會議管理系統的三個限界上下文。圖中的箭頭表示它們之間的事件數據流。
下面的列表提供了關於圖1中的箭頭的更多信息。您能夠在討論單獨限界上下文的章節中找到更多的細節。
Gary(CQRS專家)發言:
會議管理限界上下文引起的一些事件是粗粒度的,包含多個字段。請記住,會議管理是一個使用建立、讀取、更新和刪除(CRUD)模式的限界上下文,不會引起細粒度的領域事件。有關更多信息,請參見第5章:「準備發佈V1版本」。
在旅程(開發)的規劃(初期)階段,很明顯,這些是領域中的天然劃分,能夠包含各自獨立的領域模型。其中一些劃分比其餘的更容易識別。例如,很明顯,會議管理限界上下文獨立於領域的其餘部分。它具備與定義會議和座位類型相關的明肯定義的職責,以及與應用程序其餘部分的明肯定義的集成點。
另外一方面,咱們花了一些時間才意識到訂單和註冊限界上下文與支付限界上下文是分開的。例如,直到應用程序的V2發行版,當OrderPaymentConfirmed事件成爲OrderConfirmed事件時,與支付相關的全部概念才從訂單和註冊上下文中消失。
Gary(CQRS專家)發言:
隨着咱們對領域模型理解的加深,咱們在整個過程當中不斷細化領域模型。
更實際,從旅途的角度來看,咱們想要一組限界上下文,使咱們可以發佈一個可工做的應用程序幷包含一些核心的功能,它可使咱們去探索許多不一樣的實現模式:CQRS, CQRS/ES,以及與傳統的集成,例如CRUD風格的限界上下文。
Beth(業務經理)發言: Contoso但願儘快發佈一個可用的應用程序,可是還要可以在開發過程當中添加計劃的特性和客戶要求的特性,而且不須要停機就能進行升級。