不容錯過!什麼是領域驅動設計?為什麼落地這麼難?

来源:https://www.cnblogs.com/Jcloud/archive/2022/08/25/16623237.html
-Advertisement-
Play Games

引言 領域驅動設計並不是新的架構設計理論,從Eric Evans提出至今已經有十多年曆史。由於微服務架構的興起,DDD常用於指導微服務邊界劃分,並重新廣泛進入軟體研發大眾的視野。DDD的理念及應用普及在國外相對成熟,在國內尚處於初期發展階段。國內的很多社區以及企業組織內部近幾年對於DDD的探討和應用 ...


引言

領域驅動設計並不是新的架構設計理論,從Eric Evans提出至今已經有十多年曆史。由於微服務架構的興起,DDD常用於指導微服務邊界劃分,並重新廣泛進入軟體研發大眾的視野。DDD的理念及應用普及在國外相對成熟,在國內尚處於初期發展階段。國內的很多社區以及企業組織內部近幾年對於DDD的探討和應用逐漸火熱,許多架構師以及開發人員對DDD充滿了學習和實踐的熱情。而像敏捷一樣,不同團隊對其認知水平和實踐水平不盡相同,有的成功,大多數可能是失敗的。

領域驅動設計(Domain Driven Design),簡稱DDD, Eric Evans 2004年的《Domain-Driven Design: Tackling Complexity in the Heart of Software》一書中第一次提出。領域驅動設計是一種用於指導軟體設計的方法論,也是一種設計思維方式,用於解決軟體複雜性問題,旨在加速那些必須處理複雜領域的軟體項目的開發

實踐DDD的第一步不在於如何編寫代碼,而首先需要拉齊對領域驅動設計的認知。後續的系列文章將圍繞領域驅動設計進行不同視角探討,以期幫助大家對其有更深入的認識,並能應用的實際的研發工作中。

聊聊問題空間、解空間、領域模型

問題空間和解決方案空間

 


 

問題空間:Problem Space,是當前環境下業務所面臨的一系列問題和問題背後的需求,通常是業務和產品領域專家主導問題、需求的收集描述和分析。問題空間框定了我們要解決的問題的上下文,這種上下文環境不是軟體工程或是領域驅動所獨有的,而是通用的共性的元素。工程實踐必然處於某種上下文環境之下。

解決方案空間:Solution Space,解決方案空間是針對問題空間的解決方案,屬於工程實現階段,由技術專家主導方案設計。

軟體開發過程,本質上可以看作是問題空間到解決方案空間的一個映射轉化:問題空間,找出業務挑戰及其對相關需求場景用例分析解空間,通過具體的技術工具手段來進行設計實現

領域、模型和領域模型

領域:Domain

“領域”是“知識或活動的集合”,相對於軟體系統而言,領域就是軟體應用所要解決的現實問題區域。領域對應於問題空間,是一個特定範圍邊界內的業務需求的總和。

領域模型:Domain Model

抽象是一種化繁為簡的能力,是人類認識世界的利器,也是一種生物本能。在有限的腦容量的前提下,人類不可能存儲記憶所有的細節,海量信息已經超出人腦存儲限制而無法容納和有效獲取。抽象使得人類可以屏蔽無關細節信息,抽取高層的有效信息進行記憶存儲。試想,如果腦機介面技術有所突破,在人腦背後鏈接的是海量的高效的電腦集群,這種無限的存儲、計算和檢索能力的增強,“抽象能力也許會被弱化”。

模型被用來表述人們所關註的現實或想法的某個方面,本質上是一種抽象過程的產物,把與解決方案密切相關的方面抽象出來,而忽略無關細節。

聚焦在軟體工程領域,要想構建滿足需求的軟體系統,開發團隊需要軟體面向的領域所涉及的知識可能非常龐大和複雜,而模型正是解決這種信息超載問題的有效工具

對領域進行模型設計的過程就是領域建模,領域建模的目的並非是要建立一個百分之百符合“現實”的模型,理論上,我們也無法實現這種對現實的完全建模,而只能是對現實某種程度的模擬

 


 

領域建模的輸出即領域模型,領域模型是針對特定領域里的關鍵事物及其關係的可視化表現,屬於解決方案空間範疇。為了準確定義需要解決問題而構造的抽象模型,為軟體系統的構建目標統一認知,是業務功能場景在軟體系統里的映射轉化。領域模型並非領域專家頭腦中的知識,而是對這些知識進行嚴格的組織和有選擇的抽象。

同時,領域模型並非某種特殊的圖、文字或者代碼,而是他們所傳達的思想,圖、文字或代碼都可以作為模型的表示或傳達形式,但他們不是模型,而是不同維度的模型視圖

 


 

領域驅動設計

領域驅動設計強調領域模型的重要性,並通過模型驅動設計來保障領域模型與程式設計的一致。領域驅動設計首先從業務需求中提煉出統一語言,並建立領域模型指導著程式設計以及編碼實現;最後,又通過重構來發現隱式概念,並不斷解決領域領域模型相關的新問題。本質上,領域驅動設計也是從問題空間映射到解決方案空間。

領域驅動設計結合了巨集觀和微觀兩個層面的設計,分別對應於領域驅動概念中的戰略設計和戰術設計。

領域驅動設計:戰略設計

戰略設計的初衷是要保持模型的完整性,主要從下麵兩個方面來考量的:

  • 問題域方面:將問題規模進行拆解,劃分為不同類型的子域,識別出核心領域與其他子領域。
  • 解決方案層面:劃分限界上下文和上下文映射對問題域進行合理的分解,確定上下文邊界以及它們之間的關係。

領域驅動設計:戰術設計

戰略設計的初衷是要保持模型的完整性,通過戰略設計將整個軟體系統分解為多個限界上下文,然後對每個界限上下文進行戰術設計。對每個限界上下文進行戰術設計。Eric Evans提供的模型驅動設計的構造要素以及要素間的關係如下圖所示:

 


 

  • 實體:Entity,不同於通過屬性進行定義的傳統對象,實體對象通過唯一標識進行區分,且具有持續的生命周期。
  • 值對象:Value Object,值對象是具有屬性且不可變的對象,但沒有唯一標識。
  • 領域事件:Domain Event,領域事件用於記錄系統內的模型活動相關的離散事件,雖然系統內所有事件都應該能夠被跟蹤,但只有被領域專家關心的事件類型才創建領域事件。
  • 聚合:Aggregate,聚合對象是實體和值對象的聚合,聚合具有一個唯一的根,即聚合根。外部對象不再直接訪問聚合內部的單個對象或者實體,而是直接訪問聚合根,並使用聚合根將指令傳遞給對應的分組。
  • 領域服務:Domain Service,某些領域邏輯不適合分配給某個特定的實體對象,可將其這些操作封裝成領域服務
  • 資源庫:Repositories,資源庫不是配置庫,它提供一個全局介面來訪問特定聚合內部所有的實體類和值對象,應該包括創建,修改,刪除聚合內部對象的方法。
  • 工廠:Factories,工廠封裝了創建複雜對象和聚合的邏輯,對客戶端屏蔽創建的複雜性

上述DDD戰術設計的模式標識了進行設計時的一些關鍵模式,但並非說是一定要嚴格使用和遵循的,也不是遵循了所有的戰術設計模式就是符合領域驅動設計。因為,實踐DDD關鍵不在於這種戰術層面模式的落地,而是在於其巨集觀的領域驅動設計思想的遵循,比如統一語言、領域模型與代碼間的一致、子域及上下文的拆分以及映射、領域模型與技術關註點的分離等等。另外,隨著DDD的不斷發展,一些新的構建模式已經涌現,老的構造模型不一定能符合團隊研發的要求。

領域驅動設計為什麼落地這麼難?

需要魯棒的領域知識,依靠項目中領域專家的支持

  • 如果團隊中沒有熟悉應用所需領域知識的領域專家,即使具備技術再強的開發人員也無濟於事。在某些情形下,領域驅動設計需要一個或多個外部人員在整個軟體開發生命周期中扮演領域專家的角色。有些情況下,領域驅動設計需要在整個軟體開發生命周期中與外部團隊成員(充當領域專家角色)進行協作。
  • 在創新型業務中應用DDD同樣存在挑戰,由於業務模式的不確定性,業務需求變化的頻率和幅度很大,同樣也缺乏新領域的領域專家,整個業務都處於一種探索模式,很難建立起相對穩定、高可復用的領域模型。

強調不斷迭代和持續集成,對缺乏迭代經驗而偏重於瀑布模型的團隊可能導致障礙

領域驅動設計實踐依賴不斷迭代和持續集成來構建高可擴展的項目,但是這種基於迭代和持續集成的時間,在某些團隊中落地可能會存在阻礙,特別是如果過去經驗是建立在僵化的開發模型上,比如瀑布模型。

不適合偏向技術型的應用

領域驅動設計適合於具有非常高領域複雜性(業務邏輯複雜)的應用,但不適用於領域複雜性很低但技術複雜性很高的領域。DDD著重強調需要領域專家以便構造出項目依賴的統一語言和領域模型,但是如果項目的技術複雜性很高,領域能否理解是一種挑戰。當全體團隊成員沒有完全理解技術需求或限制時可能會導致問題

團隊過於重視戰術設計,導致實踐準線和原則的偏離

團隊對領域驅動設計的認知不夠,精力沒有聚焦在問題域拆分、統一語言、模型與技術關註點分離等核心原則上,而是一開始便從實現的角度觸發,過度強調戰術設計模式的落地,從而陷入無盡的技術細節之中


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

-Advertisement-
Play Games
更多相關文章
  • 6 函數 6.1 函數定義 函數可以封裝語句,然後在任何地方、任何時間執行。JavaScript中的函數使用function關鍵字聲明,主要由函數名、函數參數和函數體組成。其基本語法和聲明如下所示: 方式一: function functionName(arg0, arg1,...,argN) { ...
  • <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="wi ...
  • JavaScript函數 1.函數的語法格式 function 函數名(形參1,形參2,形參3....){ // 函數註釋,解釋說明該函數用來做什麼 函數體代碼 return 返回值 } 2.無參函數 function func1(){ console.log('hello world') } fu ...
  • 如何在 IconFont 上獲取圖標資源 阿裡巴巴矢量圖標庫網站(https://www.iconfont.cn/)上提供了非常豐富的圖標資源,包括 SVG、AI、PNG、字體圖標等格式。使用該網站提供的圖標,需要在該網站上註冊賬號並登錄。 1 搜索圖標 登錄成功後,在首頁的搜索框中搜索關鍵字,例如 ...
  • 千分位格式化 在項目中經常碰到關於貨幣金額的頁面顯示,為了讓金額的顯示更為人性化與規範化,需要加入貨幣格式化策略。也就是所謂的數字千分位格式化。 例如 123456789 => 123,456,789 、 123456789.123 => 123,456,789.123 const formatMo ...
  • 根據`業務功能`進行模塊化一直以來都是後端的普遍做法,而Web前端則通常都是按照UI界面的視圖區塊`View`來進行模塊化,這樣的模塊實際上只是`Component組件`,不具備獨立自治的能力。究其原因我想是因為在早期Web1.0的時代,前端的職能就是僅僅作為後端API數據的一個Render渲染器,... ...
  • 首先MDN官網中有一句寫道: assign()語法: Object.assign(target, ...sources) target:目標對象,接收源對象屬性的對象,也是修改後的返回值。sources:源對象,包含將被合併的屬性。下麵來看例子: let s = { a: 1 } let targe ...
  • Sass提供了許多內置模塊,其中包含有用的函數(以及mixin)。這些模塊可以像任何用戶定義的樣式表一樣使用@use規則載入,它們的函數可以像任何其他模塊成員一樣調用。所有內置模塊URL都以sass開頭:表示它們是sass本身的一部分。 常見函數簡介,更多函數列表可看:https://sass-la ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...