我們有時候也會看到一些博客看到或者聽到一些同事在說:這個業務有什麼難的,不就是CRUD麽?在軟體生命周期初期,我們通過CRUD這種方式我們可以快速的實現業務規則,交付項目,但隨著業務逐漸複雜,通過CRUD這種粗暴方式不可避免地會淹沒業務核心規則,產生很多祖傳(屎山)代碼,系統交接的時候我們經常會聽到... ...
一、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,有可能不是你做的業務沒價值,而是自己對於業務的理解還不夠深;如果總是沉迷於代碼編寫,可能你的發展空間就會受限了。
作者:京東科技 孫黎明
來源:京東雲開發者社區 轉載請註明來源