DDD 領域驅動設計-商品建模之路

来源:http://www.cnblogs.com/xishuai/archive/2016/12/09/ddd-product-design.html
-Advertisement-
Play Games

最近在做電商業務中,有關商品業務改版的一些東西,後端的架構設計採用現在很流行的微服務,有關微服務的簡單概念: 微服務是一種架構風格,一個大型複雜軟體應用由一個或多個微服務組成。系統中的各個微服務可被獨立部署,各個微服務之間是松耦合的。每個微服務僅關註於完成一件任務並很好地完成該任務。在所有情況下,每 ...


最近在做電商業務中,有關商品業務改版的一些東西,後端的架構設計採用現在很流行的微服務,有關微服務的簡單概念:

微服務是一種架構風格,一個大型複雜軟體應用由一個或多個微服務組成。系統中的各個微服務可被獨立部署,各個微服務之間是松耦合的。每個微服務僅關註於完成一件任務並很好地完成該任務。在所有情況下,每個任務代表著一個小的業務能力。

關於改版的業務設計,還是想嘗試 DDD 領域驅動設計,之前寫的一些相關文章,都是直接進行戰術設計,而非在戰略設計基礎上進行,所以最後可能會出現一些問題,所以這次的過程是:邊瞭解業務、邊瞭解 IDDD 書中關於戰略設計的部分,然後嘗試使用戰略設計的方式進行業務分析,最後再細分出具體的戰術設計,沒有正確的設計方案,只有合適的設計方案,排除技術之外的業務分析過程,還是蠻有意思的。

DDD 戰略建模(包含概念):領域(Domain)、核心域、子域、界限上下文(Bounded Context)、上下文映射圖(Context Mapping)。

相關文章:

1. 業務流程

業務場景:發佈商品

業務場景就上面四個字,看起來很簡單,但其實具體分析起來,所包含的東西還是蠻多的,整個發佈商品過程,就像一個商品誕生的生命周期一樣,需要經歷各個階段和過程,直到商品正式發佈出來,並且在這個過程中,有一系列的其他概念由商品衍生出來,比如庫存、分類、品牌等等,還會有一些用戶的行為參與,比如小二的後臺審核等。

在業務分析過程中,我還是比較喜歡畫一張簡單的業務流程圖,並不一定很規範,你也可以直接手繪出來,從業務流程圖中,我們可以看到整個的業務方向,有利於我們從中找出關鍵的業務點,併進行具體分析設計。

發佈商品業務流程圖:

圖比較簡單,我們需要從添加商品到發佈商品完成的過程中,抽離出關鍵的業務點,並且這些業務點事需要在業務系統中進行設計的,發佈商品流程大概分為兩個部分:

  1. 商戶發佈商品:這部分內容比較多,先選擇分類(分類需要進行設計),然後填寫基本信息(根據實際的業務,有很多不同的設計,是發佈商品的核心,需要重點考慮),填寫完成之後(兩種選擇:保存草稿和發佈),在小二審核之前,需要填寫入庫單,用來更新商品庫存,然後進入小二審核階段,如果審核成功,並且商戶選擇商品上架,則代表著整個商品發佈的業務流程跑完了。
  2. 小二審核商品:小二根據一些規定進行審核商戶發佈的商品,這個部分人工因素很大,但發佈商品的規定一般是確定,因為是人工進行操作,所以這部分內容在業務系統設計方面體現不大。

其實,選擇分類可以歸納到填寫基本信息中,重要的是基本信息具體是什麼?這部分包含的業務是什麼?該如何設計呢?後來分析了下,除了一些商品的基本信息之外(比如標題、價格、商品詳情、商品圖片),還包含了品牌、分類、屬性(一般指的是 sku)等,像標題和價格之類的屬性一般是具體的值,後面我們在戰術設計的時候,直接把它們設計成值對象即可,但對於品牌、分類之類的對象,需要進行單獨進行設計,因為它們不是一個值所能代表的,需要進行獨立維護。

除了商品信息之外,後面就是填寫採購單用來更新商品庫存了,這部分業務內容有點像外部服務一樣,通過外部服務的一些操作,最後的結果導向商品模型,這部分類似的業務以後可能會很多,比如小二審核商品信息,也像一個外部服務一樣,不過是人為進行操作的,審核最後的結果導向商品狀態,我們可以歸納出,可以改變商品狀態的一些行為,都是需要進行考慮的業務,並且這部分業務在後面建模的時候,需要重點設計。

畫業務流程圖的目的,在於熟悉整個業務的大致流程,以及對商品生命周期的瞭解,但只是大致的表述,當你對業務理解越深的時候,業務流程圖也就會越複雜,但基本的框架是不變的,所以,在畫的業務流程圖的時候,要找出業務的不變規則,比如發佈商品肯定要填寫信息、然後小二審核等,變的業務都是在這些不變的規則之上豐富起來的,最後形成整個健全的業務系統。

2. 限界上下文

關於領域、核心域和子域的概念,相對比較容易理解,領域就是業務系統的全部,核心域就是業務系統最重要的部分,比如商品業務系統,核心域就是商品,其他相對不重要的業務部分就是子域,子域又分為支撐子域和通用子域,支撐子域用來支撐核心域,在整個領域中,可以被公用的子域,稱為通用子域。

限界上下文是一個顯式的邊界,領域模型存在這個邊界職位,領域模型把通用語言表達成軟體模型,一般在設計的時候,會把領域和限界上下文一一對應(但也不是相對的),有時候限界上下文很大,但有時候限界上下文也很小,比如一段業務描述也可以稱之為限界上下文,不管概念是怎麼定義的,只需要知道限界上下文的核心是邊界,邊界的目的就是內聚合隔離。

一張簡單的商品限界上下文圖(虛線表示領域的邊界):

首先,在商品領域中,商品是核心域,並對應一個商品上下文,庫存被設計為一個通用子域,因為以後交易的業務場景會被用到,並對應一個庫存上下文,品牌通用子域也一樣,業務場景可能會對品牌的單獨處理(比如品牌街,這是和商品不想關的),所以設計成通用子域會相對好些,分類支撐子域和屬性子域相對複雜點,其實這裡的分類和屬性都是相對於商品而言的,你可以成為商品分類和商品屬性,獨立於商品之外,分類和屬性是沒有任何存在的業務意義的,所以,把它們設計為商品領域的支撐子域會比較好些。

另外,關於分類上下文和屬性上下文之間的關係,從上面圖中就可以看到,在業務場景中屬性依附於分類,比如在發佈商品頁面,填寫商品屬性之前需要先確定商品分類,因為不同的分類有對應不同的屬性,比如錶帶材質屬性,只有手錶分類下才會有,其實它們也可以直接合二為一,叫做分類屬性支撐子域,對應分類屬性上下文,上面說過添加屬性之前,必須先確定分類,屬性就像是分類中的一個子域,屬性其實和商品沒有直接的關係,它和商品的所有關係,必須都通過分類,並且屬性的數據維護也是如此,這個後面會有調整,再詳細說明。

分類在具體的實現中,會相對比較簡單,有點像品牌的實現,頂多和商品有一些關聯,但屬性實現相對比較複雜些,因為屬性項和屬性值都是動態的,並且屬性的展現形式也是動態的,比如一個屬性項可能對應多個屬性值,並且展現可能是組合形式的(文本+單選+下拉列表),這方便在也體現在數據存儲的時候,不過可以按照一定的格式用 json 進行存儲,展現方式也是一樣。

限界上下文就像一個手術刀,將領域一點一點的進行解剖,解剖出來的部位獨立進行實現,限界上下文的具體實現就是戰術設計,並且各個限界上下文的實現之間是相互不影響。

3. 數據模型(聚合和實體)

限界上下文讓我們明白,我們到底需要做什麼東西,接下來就是針對這些東西的具體設計和實現了,怎麼實現?就是戰術設計。

戰術建模(包含概念):聚合(Aggregate)、實體(Entity)、值對象(Value Objects)、資源庫(Repository)、領域服務(Domain Services)、領域事件(Domain Events)、模塊(Modules)。

關於戰術設計的首要前提是聚合,後面實體和值對象等概念,都是在聚合的基礎上衍生出來的,關於聚合的概念就不多說了,但需要說明下聚合設計的註意點:

  • 儘量小聚合設計:有助於減少事務的提交衝突,也有利於系統性能和可伸縮性,但不能過小,比如一個聚合只包含唯一標識和單個屬性,這種設計不合理。
  • 根實體可以作為聚合根。
  • 聚合邊界和真實的業務約束是一致的。
  • 通過唯一標識引用其他聚合。
  • 事務一致性:在一個事務中,只能修改一個聚合實例。
  • 在聚合之外使用最終一致性:如果可以不在意延遲,一般用領域事件進行實現。

先簡單看下商品所包含的東西:

圖中主要說明的是商品大致包含的內容:分類、屬性和基本信息,屬性又有具體的分類,但都基於分類確定的情況下。

一張簡單的商品數據模型圖:

簡單歸納下:

  • 商品(聚合根):商品(根實體)、商品圖片(實體)、商品 sku(實體)、商品描述(實體)、商品調整紀錄(實體)
  • 庫存(聚合根):庫存(根實體)、入庫詳情(實體)
  • 品牌(聚合根):品牌(根實體)
  • 分類(聚合根):分類(根實體)、分類屬性(實體)、分類屬性值(實體)

實體中的屬性只是一些示例,並不詳細,值對象並沒有在圖中體現,因為實體的屬性都可以被設計為值對象,這部分在具體實現的時候,再詳細進行考慮,聚合根和實體、聚合根和聚合根之間的關係用箭頭進行表示了。

關於分類和屬性,在限界上下文設計的時候,被分開設計了,但後來想了一下,還是設計成一個比較好,分類作為聚合根,分類屬性和分類屬性值作為衍生出來的實體。

關於數據模型圖,就不詳細說明瞭,內容都在上面的圖中,況且現在還不是很完善,後面可能還會進行調整。

大概就紀錄這些。



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

-Advertisement-
Play Games
更多相關文章
  • 如果在調用PHP查詢資料庫,在echo後返回的是Resource id #9,可能你的輸出方式是: 按以上方式直接輸出,由於mysql_query($sql)本身返回的就是一個resource,也就是返回的就是Resource id #9,因此在輸出之前,還需要加一個函數: ...
  • 電腦程式的思維邏輯 (1) - 數據和變數 電腦程式的思維邏輯 (2) - 賦值 電腦程式的思維邏輯 (3) - 基本運算 電腦程式的思維邏輯 (4) - 整數的二進位表示與位運算 電腦程式的思維邏輯 (5) - 小數計算為什麼會出錯? 電腦程式的思維邏輯 (6) - 如何從亂碼中恢復 ...
  • 一、異常處理: 1 AttributeError 試圖訪問一個對象沒有的樹形,比如foo.x,但是foo沒有屬性x 2 IOError 輸入/輸出異常;基本上是無法打開文件 3 ImportError 無法引入模塊或包;基本上是路徑問題或名稱錯誤 4 IndentationError 語法錯誤(的子 ...
  • ...
  • 好久沒寫了,都忘記博客了,趁著現在還在公司,寫的東西是經過驗證的,不是在家憑記憶力寫的,正確率有保障,就說說最近遇到的一件事情吧。 以前一直用的oracle資料庫,這次項目我負責的模塊所在的系統是用的mysql資料庫,結果當初建表時候,欄位什麼的全靠百度,實在是英語不行,然後有個欄位叫usage,是 ...
  • 1、AcceptEx() AcceptEx()用於非同步接收連接,可以取得客戶程式發送的第一塊數據。 [cpp] view plaincopy BOOL AcceptEx( _In_ SOCKET sListenSocket, //監聽套接字句柄 _In_ SOCKET sAcceptSocket, ...
  • AJAX 是與伺服器交換數據的藝術,它在不重載全部頁面的情況下,實現了對部分網頁的更新。編寫常規的 AJAX 代碼並不容易,因為不同的瀏覽器對 AJAX 的實現並不相同。這意味著您必須編寫額外的代碼對瀏覽器進行測試。不過,jQuery 團隊為我們解決了這個難題,我們只需要一行簡單的代碼,就可以實現 ...
  • 1,super關鍵字 super:父類的意思 1. super.屬性名 (調用父類的屬性) 2. super.方法名 (調用父類的方法) 3. super([參數列表])(調用父類的構造方法) 註意:a. super關鍵字使用在子類中 b. 子類繼承了父類,則父類中的公有的屬性,方法,就是子類的屬性 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...