DDD學習與感悟——總是覺得自己在CRUD怎麼辦?

京東雲開發者發表於2023-12-04

一、DDD是什麼?

DDD全名叫做Domins drives Design;領域驅動設計。再說的通俗一點就是:透過領域建模的方式來實現軟體設計。

問題來了:什麼是軟體設計?為什麼要進行軟體設計?

軟體開發最主要的目的就是:解決一個問題(業務)而產生的一個交付物(系統)。而軟體設計旨在高效的實現複雜專案軟體。也就是說軟體設計是從業務到系統之間的橋樑。

而DDD則是在複雜業務場景下一種更高效更合理的軟體設計思維方式和方法論。

二、以前的軟體設計思維是什麼?

絕大部分從事軟體開發的人,不管是在學校還是剛開始工作,都是從ER圖開始。即直接透過業務設計資料庫模型和資料關聯關係。這種思維根深蒂固的印在了這些人的頭腦裡(包括我自己)。因此在軟體設計過程中習慣性的直接將業務轉化為資料模型,面向資料開發。也就是我們所說的CRUD。我們有時候也會看到一些部落格看到或者聽到一些同事在說:這個業務有什麼難的,不就是CRUD麼?

不可否認的是,在軟體生命週期初期,透過CRUD這種方式我們可以快速的實現業務規則,交付專案。然而一個系統的生命週期是很長的並且維護階段的生命週期佔絕大部分比例。 隨著業務的發展,業務規則越來越複雜,透過CRUD這種粗暴方式,讓工程程式碼越來越複雜,通常一個方法可能會出現幾百甚至上千行程式碼,各種膠水程式碼和業務邏輯混合在一起,導致很難理解。

這種系統交接給另一個同學或者新進來的同學後,可能需要花費很長的時間才能理解這個方法,原因就是因為這種膠水程式碼淹沒了業務核心規則。所以在現實場景中,我們經常會聽到,上一個開發是SB,或者自嘲自己是在屎山上面繼續堆屎。

三、DDD思想下的軟體設計

DDD的思想是基於領域模型來實現軟體設計。那麼,什麼是領域模型?領域模型怎麼得來呢?

DDD思想,將軟體的複雜程度提前到了設計階段。基於DDD思想,我們的設計方式完全變了。

統一語言

首先,將業務方、領域專家以及相關的產研人員都聚攏在一起,共同探討出業務場景和要解決的問題,統一語言。來確保所有人對於業務的理解都是一致的。

這裡的統一語言不是指某種具體的技術語言,而是一種業務規則語言。所有人必須要能夠理解這種統一語言。

戰略設計

其次,我們根據待解決的問題空間,進行戰略設計。所謂的戰略設計就是根據問題空間在宏觀層面識別出限界上下文。比如說一個電商業務,我們需要交付一個電商系統,根據電商業務的特點,需要劃分出使用者、商品、訂單、倉儲等限界上下文,每一個限界上下文都是一個獨立的業務單元,具有完整的業務規則。

識別領域模型

然後,再分別針對上下文內的業務領域進行建模,得到領域模型。在DDD思想中,領域模型中通常包含實體、值物件、事件、領域服務等概念。我們可以透過“事件風暴”的方式來識別出這些概念。

注意,“事件風暴”和“頭腦風暴”是有區別的。“頭腦風暴”的主要目的是透過發散思維進行創新,而“事件風暴”是DDD中的概念,其主要目的是所有人一起根據統一語言和業務規則識別出事件。再根據事件識別出實體、值物件、領域服務、指令、業務流等領域模型中的概念。

所謂事件指的是已經發生了的事情。比如使用者下了一個訂單、使用者取消了訂單、使用者支付了訂單等

根據事件,我們可以識別出實體,比如上面這個例子中的訂單實體,以及指令:取消、支付、下單等。

程式設計

識別出領域模型之後,我們就可以根據領域模型來指導我們進行程式設計了。這裡的程式設計包括業務架構、資料架構、核心業務流程、系統架構、部署架構等。需要注意的是,在進行程式設計時,我們依然要遵循DDD中的設計規範。否則很容易走偏方向。

編寫程式碼

有了完整的程式設計之後,我們就可以進行實際的工程搭建以及程式碼編寫了。

這個階段需要注意的是,我們需要遵循DDD思想中的架構設計和程式碼設計。實際上這個階段也是非常困難的。因為基於DDD思想下的工程架構和我們傳統的工程架構不一樣。

基於DDD思想下,編碼過程中我們經常會遇到的一個問題是:這個程式碼應該放在哪裡合適。

工程結構

在DDD中,標準的工程結構分為4層。使用者介面層、應用層、領域層和基礎設施層。

DDD中,構建軟體結構思維有六邊形架構、CQRS架構等,它們是一種思想,是從邏輯層面對工程結構進行劃分,而我們熟知的SOA架構以及微服務架構是從物理邏輯層面對工程結構進行劃分,它們有著本質的區別,但是目標都是一樣的:構建可維護、可擴充套件、可測試的軟體系統。

程式碼編寫

在DDD中,最為複雜的便是領域層,所有的業務邏輯和規則都在這裡實現。因此我們經常會遇到一個問題就是程式碼應該放在哪裡。

在具體落地過程中會遇到這些問題,解決這些問題沒有銀彈,因為不同的業務有不同的處理方式,這個時候我們需要與領域專家們討論,得出大家都滿意的處理方案。

程式碼重構

沒有不變的業務。因此我們需要結合業務的發展而不斷迭代更新我們的領域模型,透過重構的方式來挖掘隱形概念,再根據這些隱形概念去不斷的調整我們的戰略設計以及領域模型。使得整個軟體系統的發展也是螺旋式迭代更新的過程。

透過以上的介紹,我們實現DDD的過程如下:

四、總結

透過對於DDD的理解,其實不難發現,程式設計師的工作重心變了,程式設計師其實不是在編寫程式碼,而是在不斷的摸索業務領域知識,尤其是複雜業務。

所以如果總是覺得自己在CRUD,有可能不是你做的業務沒價值,而是自己對於業務的理解還不夠深;如果總是沉迷於程式碼編寫,可能你的發展空間就會受限了。

作者:京東科技 孫黎明

來源:京東雲開發者社群 轉載請註明來源

相關文章