亚洲视频二区_亚洲欧洲日本天天堂在线观看_日韩一区二区在线观看_中文字幕不卡一区

公告:魔扣目錄網(wǎng)為廣大站長提供免費收錄網(wǎng)站服務(wù),提交前請做好本站友鏈:【 網(wǎng)站目錄:http://www.430618.com 】, 免友鏈快審服務(wù)(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747

作者 | 劉嘿嘿、離夏、立羽

導(dǎo)讀

introduction

最近幾年,微服務(wù)拆分大行其道,在業(yè)務(wù)越來越復(fù)雜的情況下,許多業(yè)務(wù)紛紛拋棄了傳統(tǒng)單體架構(gòu),擁抱微服務(wù)。但隨著微服務(wù)的拆分結(jié)束,大家又發(fā)現(xiàn)了新的問題,比如服務(wù)間邏輯復(fù)雜,運維復(fù)雜性變高,微服務(wù)架構(gòu)變得越來越難以管理,最終演化成大泥球架構(gòu)。

而本文主要介紹如何通過DDD對微服務(wù)進行拆分,首先介紹了什么是DDD,通過從分析DDD的優(yōu)勢,到如何通過DDD進行業(yè)務(wù)拆分,并且在最后通過代碼樣例的方式,深入淺出的為讀者介紹了DDD代碼的核心實現(xiàn)。幫助大家進一步的了解DDD應(yīng)該如何落地。

全文6271字,預(yù)計閱讀時間16分鐘。

GEEK TALK

01

什么是DDD

DDD(領(lǐng)域驅(qū)動設(shè)計),起源于2004年Eric Evans出版《領(lǐng)域驅(qū)動設(shè)計》,近些年由于微服務(wù)的興起,大家逐漸對單體服務(wù)進行拆分。

但是隨著微服務(wù)拆分,由于業(yè)務(wù)邏輯拆分不合理導(dǎo)致調(diào)用環(huán)路問題、重試風(fēng)暴問題等等,都給系統(tǒng)造成了更多的風(fēng)險,并且隨著業(yè)務(wù)更加復(fù)雜微服務(wù)職責(zé)劃分出現(xiàn)問題,則業(yè)務(wù)迭代效率變得越來越差,最終變成一個大泥球系統(tǒng)。

而DDD的優(yōu)勢便是指導(dǎo)業(yè)務(wù)進行微服務(wù)拆分,下面我們以會員中心為例來具體講解一下如何進行業(yè)務(wù)拆分以及相關(guān)的代碼實現(xiàn)。

GEEK TALK

02

使用DDD的優(yōu)勢是什么

2.1 語言統(tǒng)一,消除誤解

很多時候未必產(chǎn)品經(jīng)理才是最懂業(yè)務(wù)的那個人,例如某些B端服務(wù)很多時候是運營人員在向產(chǎn)品同學(xué)提需求,在經(jīng)過產(chǎn)品經(jīng)理的翻譯后,才轉(zhuǎn)化成一個需求文檔,這樣就會導(dǎo)致有時候產(chǎn)品經(jīng)理并不能完全表達出實際的需求,這就會導(dǎo)致開發(fā)人員交付的軟件無法達到預(yù)期。從而導(dǎo)致返工,浪費人力。

而DDD需要設(shè)計一種通用的語言,拉齊各個需求方的理解,一旦產(chǎn)品同學(xué)和技術(shù)同學(xué)對業(yè)務(wù)具備了相同的理解,統(tǒng)一的語言,那在后續(xù)的需求迭代種就會變得非常順暢。

在改造初期我們耗費了非常大的精力向產(chǎn)品同學(xué)講清楚哪些抽象應(yīng)該定義為實體,實體與實體的關(guān)系是什么,在不斷的溝通、磨合中,最好我們成功建立起了一些通用的語言,拉齊了產(chǎn)品經(jīng)理、運營同學(xué)、開發(fā)人員的理解,最大幅度的消除了由于理解不一致導(dǎo)致的返工、重構(gòu)等工作。

2.2 更專注于業(yè)務(wù)的戰(zhàn)略設(shè)計

戰(zhàn)略設(shè)計側(cè)重于業(yè)務(wù)梳理,結(jié)合業(yè)務(wù)流程劃分對應(yīng)的核心域、通用域、支撐域。戰(zhàn)略設(shè)計的核心價值是圍繞產(chǎn)品規(guī)劃重點投入資源,確保重點子業(yè)務(wù)可以確保得到足夠的人力支持。

2.3設(shè)計即代碼,代碼即設(shè)計

在過去的項目詳細設(shè)計中,我們的重心在數(shù)據(jù)怎么存儲?數(shù)據(jù)流通是什么樣的。這樣可能導(dǎo)致在設(shè)計文檔和代碼中就具備較大的Gap,實現(xiàn)上就可能有問題。

而DDD倡導(dǎo)的是思考,而不是寫代碼。在代碼設(shè)計之前定義好領(lǐng)域語言,和領(lǐng)域?qū)<覝贤o礙,定義好領(lǐng)域規(guī)則,這樣在寫代碼的時候留下較少的思考。代碼只是把設(shè)計文檔翻譯成代碼,寫代碼更像是在照著設(shè)計文檔在做填空題,只需要將代碼填到指定的文件中即可。

GEEK TALK

03

如何使用DDD

3.1 DDD戰(zhàn)略設(shè)計

3.1.1 劃分核心域,通用域、支撐域

在實際的工作中,很多產(chǎn)品經(jīng)理會陷入到各種繁雜的業(yè)務(wù)指標中,無法從繁雜的業(yè)務(wù)中抽身,定義好哪些是重要的模塊,或者無法表達出業(yè)務(wù)各個模塊中最重要的是什么。這種情況就會導(dǎo)致每個,產(chǎn)品沒有這就會導(dǎo)致在人員分工、資源申請上出現(xiàn)一些問題。

做戰(zhàn)略設(shè)計,最核心的事情就是劃分清楚核心域,通用域、支撐域,我們把更多的精力投入到核心的問題中,而不被大量次要的問題淹沒。

 

  • 核心域:業(yè)務(wù)最核心的部分,這部分需要產(chǎn)品同學(xué)確定,例如,從長線來看我們主要核心做的投入,是做流量引入,還是做變現(xiàn)
  • 支撐域:業(yè)務(wù)中非核心的部分,若產(chǎn)品確定現(xiàn)有核心域是流量引入,那在流量變現(xiàn)部分業(yè)務(wù),就是支撐域
  • 通用域:例如登錄驗證、驗證碼、支付能力等則更多的使用公司內(nèi)部的中臺能力,若公司沒有通用的中臺能力,我們也會以建設(shè)中臺的思路自建一個內(nèi)部的中臺服務(wù)
本處僅僅描述我們對于戰(zhàn)略設(shè)計理解,不對戰(zhàn)略設(shè)計展開說明。

 

3.1.2 劃分邊界

微服務(wù)職責(zé)的劃分是執(zhí)行環(huán)節(jié)的第一步,也是最重要的一步,尤其從大單體拆分為多個微服務(wù)時,需要考慮以下幾點

 

  1. 要通過領(lǐng)域驅(qū)動劃分邊界,若暫時考慮不清楚邊界,那就先不要拆分
  2. 明確微服務(wù)分層,上游服務(wù)只能對下游服務(wù)產(chǎn)生依賴,防止微服務(wù)環(huán)路調(diào)用問題,同時下游服務(wù)需要考慮重試風(fēng)暴問題
  3. 核心域的微服務(wù)需要具備故障降級,容災(zāi)能力
  4. 要基于組織架構(gòu)進行邊界的劃分,微服務(wù)的梳理其實也是團隊的梳理,過度的拆分可能導(dǎo)致更多的溝通成本

 

3.2 DDD戰(zhàn)術(shù)設(shè)計

3.2.1 名詞解釋

聚合與聚合根:是一組相關(guān)對象的組合,可以作為拆分微服務(wù)的最小單位,具有高內(nèi)聚、低耦合的特點,聚合在DDD中是一個很重要的概念,核心領(lǐng)域往往都需要用聚合來表達;聚合根為其根節(jié)點,聚合根有實體的特點,具有全局唯一標識,有獨立的生命周期。一個聚合只有一個聚合根,聚合根在聚合內(nèi)對實體和值對象采用直接對象引用的方式進行組織和協(xié)調(diào),聚合根與聚合根之間通過 ID 關(guān)聯(lián)的方式實現(xiàn)聚合之間的協(xié)同。

領(lǐng)域服務(wù):一些重要的領(lǐng)域行為或操作,可以歸類為領(lǐng)域服務(wù)。它既不是實體,也不是值對象的范疇。

領(lǐng)域事件:領(lǐng)域事件是對領(lǐng)域內(nèi)發(fā)生的活動進行的建模。

實體:多個屬性、行為及操作的載體,實體有全局唯一性標識(ID),有獨立的生命周期。例如會員用戶中,每個會員都可以被認為一個實體,都有userid唯一性標識。

值對象:通過對象屬性來識別的對象,沒有標識符概念,無生命周期,只描述業(yè)務(wù)屬性。如在一個會員系統(tǒng)中,會員權(quán)益信息集合即可看為一個值對象,只用于對權(quán)益屬性的描述,只有數(shù)據(jù)初始化操作和有限的不涉及修改數(shù)據(jù)的行為。

3.2.2 如何進行戰(zhàn)術(shù)設(shè)計

接下來我們以會員中心為例為大家詳細介紹

在戰(zhàn)略模型中我們已經(jīng)劃分清楚邊界,梳理不同領(lǐng)域及相關(guān)關(guān)系。接下來我們需要從戰(zhàn)術(shù)層面上剖析領(lǐng)域模型內(nèi)部之間的關(guān)系,對會員上下文進行建模(下文為簡化版)。


 

在會員上下文中,我們以會員實體為中心,通過會員(vipinfo)這個聚合根來控制會員權(quán)限,一個會員包括用戶ID(uid)、會員權(quán)益(Privilege)、所屬機構(gòu)(tp)以及會員碼(vip_code),而會員碼針對訂單維度分別對應(yīng)不同的權(quán)益內(nèi)容(privilege)。

這些值對象不具有業(yè)務(wù)行為特征,只關(guān)心本身屬性值。會員實體具有業(yè)務(wù)行為及業(yè)務(wù)邏輯,例如會員入駐、會員變更、會員綁碼等,外部訪問會員權(quán)益值對象等都需要通過會員實體來進行。

在會員域中,我們同時支持會員碼及會員維度的領(lǐng)域服務(wù),包括購買、獲取會員信息、綁碼等服務(wù)。

3.3 DDD代碼實現(xiàn)

3.3.1 項目介紹

  • 會員微服務(wù)主要實現(xiàn)獲得會員信息、會員碼信息、綁會員碼、會員碼退款等操作。
  • 服務(wù)使用ddd四層架構(gòu),分為接口層、應(yīng)用層、領(lǐng)域?qū)雍突A(chǔ)層。
  • 本服務(wù)因為業(yè)務(wù)復(fù)雜性較低,為減少冗余代碼,使用松散分層。(架構(gòu)根據(jù)耦合的緊密程度又可以分為兩種:嚴格分層架構(gòu)和松散分層架構(gòu)。嚴格分層:任何層只能依賴與他相鄰的下層。松散分層:任何層可以依賴任意他的下層。)

 

3.3.2 項目結(jié)構(gòu)

項目結(jié)構(gòu)如圖所示分為四層、對應(yīng)到到代碼目錄上(附錄1),代碼一級目錄有interface(接口層)、Application(應(yīng)用層)、domAIn(領(lǐng)域?qū)樱nfrastructure(基礎(chǔ)層)四個目錄。


 

接口層:

接口層處理接口定義、批處理相關(guān)邏輯。目錄如下:

|-- interface | |-- command // 批處理接口層 | | |-- controller | | | `-- vip | | | |-- add.go | | | `-- update.go | | |-- router.go // 代碼入口定義 | | `-- script.go | `-- http // api接口層 | |-- controller // 接口入?yún)⑿r灐⒍x,調(diào)用下層代碼 | | |-- lawyer | | | |-- add.go | | | `-- update.go | | `-- vipcode | | |-- add.go | | `-- update.go | |-- router.go // api路由

interface目錄下有command、http兩個目錄,其中,

command:包含批處理入口,批處理路由,編排批處理相關(guān)領(lǐng)域?qū)臃?wù)、事件、實體和基礎(chǔ)層相關(guān)函數(shù)。批處理代碼無需應(yīng)用層直接依賴領(lǐng)域?qū)印⒒A(chǔ)層,降低代碼冗余度。

http:包含接口路由、定義,接口入?yún)⑿r灐⒍x。

應(yīng)用層:

主要負責(zé)組織、編排領(lǐng)域?qū)臃?wù)、事件、實體和基礎(chǔ)層相關(guān)函數(shù)。

application下有service、viewmodel。

|-- application | |-- service //應(yīng)用層服務(wù) | | |-- lawyer | | | |-- add.go | | | `-- update.go | | `-- vip | | |-- add.go | | `-- update.go | `-- viewmodel // 視圖 | |-- lawyer | | |-- transform.go // 轉(zhuǎn)化函數(shù) | | `-- vm.go //視圖數(shù)據(jù)結(jié)構(gòu) | `-- vip | |-- transform.go | `-- vm.go

service:對多個領(lǐng)域服務(wù)、基礎(chǔ)層ral調(diào)用、數(shù)據(jù)持久化服務(wù)進行封裝、編排,為上層提供更粗粒度的服務(wù),調(diào)用領(lǐng)域?qū)臃?wù),倉儲和事件,因為松散分層結(jié)構(gòu),也可以調(diào)用基礎(chǔ)層服務(wù)。

viewmodel:為上層多變的數(shù)據(jù)結(jié)構(gòu)要求,提供相應(yīng)視圖定義和實體到視圖的轉(zhuǎn)化方法。

領(lǐng)域?qū)樱?/p>

領(lǐng)域?qū)哟娣艠I(yè)務(wù)核心邏輯包括聚合根、實體、值對象、倉儲接口、領(lǐng)域服務(wù)、領(lǐng)域事件接口等。

領(lǐng)域?qū)酉路钟形鍌€目錄:

|-- domain | |-- aggregate // 聚合 | | |-- lawyer | | | |-- entity.go // 實體定義 | | | `-- vo.go // 值對象 | | `-- vipcode | | |-- entity.go | | |-- vo.go | |-- event // 領(lǐng)域事件 | | `-- vipcode | | `-- order.go | |-- repository // 倉儲接口 | | |-- lawyer.go | | `-- vipcode.go | |-- adaptor // 防腐層 | | `-- sms.go | `-- service // 領(lǐng)域服務(wù) | |-- lawyer | | `-- vipcode.go | `-- vipcode | `-- vipcode.go

aggregate:放置聚合根,實體、值對象數(shù)據(jù)結(jié)構(gòu)定義,以及相關(guān)初始化代碼。

領(lǐng)域內(nèi)數(shù)據(jù)流轉(zhuǎn)處理依賴,相關(guān)聚合根,下游服務(wù)發(fā)生改變——如數(shù)據(jù)表結(jié)構(gòu)變換,只需將相關(guān)數(shù)據(jù)轉(zhuǎn)化為業(yè)務(wù)定義聚合根,代碼更改只需在基礎(chǔ)層,不涉及上層。

下面是會員碼實體示例,里面又包含有訂單值對象,會員碼機構(gòu)值對象和會員碼權(quán)益值對象。

// EntityVipCode 會員碼實體(簡化版本) type EntityVipCode struct { ValidityStart *time.Time // 綁碼開始時間 ValidityEnd *time.Time // 綁定會員碼結(jié)束時間 OrderInfo *VOOrderInfo // 訂單信息值對象 BuyerInfo *VOCodeBuyerInfo // 買會員碼機構(gòu)信息 PrivilegeInfo *VOPrivilege // 會員碼包含的權(quán)益 }

event:放置基礎(chǔ)層事件抽象的接口——為了實現(xiàn)依賴倒置。

repository:放置基礎(chǔ)層數(shù)據(jù)持久化服務(wù)抽象的接口。

service:存放一下領(lǐng)域服務(wù)代碼,向應(yīng)用層服務(wù)提供方法調(diào)用,依賴倒置在ddd中使用頻繁 。

adaptor:存放防腐層數(shù)據(jù)結(jié)構(gòu)定義、轉(zhuǎn)化函數(shù)。

防腐層在下游服務(wù)和上游服務(wù)之間,將下游服務(wù)翻譯為上游服務(wù)語言,拋去無需關(guān)注的,防止上層服務(wù)摻雜過多無需關(guān)注雜質(zhì)。

ddd中廣泛應(yīng)用了依賴倒置原則(即調(diào)用要依賴于抽象接口,不要依賴于具體實現(xiàn)),減少ddd各層之間的耦合性,提高系統(tǒng)的穩(wěn)定性,減少并行開發(fā)風(fēng)險,提高代碼的可讀性和可維護性,非常適合ddd這樣為應(yīng)對頻繁迭代的設(shè)計思想。

如下創(chuàng)建訂單體現(xiàn)依賴倒置思想,無需關(guān)注具體實現(xiàn),假若使用訂單方發(fā)生了變更(如更換服務(wù)提供方、服務(wù)提供方更換實現(xiàn)邏輯或者服務(wù)實現(xiàn)邏輯更改了),我們在上層代碼只需要更改傳入的參數(shù),無需關(guān)注其他變更。

// ReqCreateOrder 創(chuàng)建訂單 func ReqCreateOrder(ctx context.Context, vipRepo repository.IVipCodeRepo, vipcodeentity vipcode.EntityVipCode) (*order.PreorderRetData, error) type IVipCodeRepo interface { CreateOrder(ctx context.Context, ev vipcode.EntityVipCode) (*liborder.PreorderRetData, error) UpdateVipCode(ctx context.Context, patch map[string]interface{}, conditions map[string]interface{}) (int64, error) }

基礎(chǔ)設(shè)施層:

基礎(chǔ)層存放領(lǐng)域事件、數(shù)據(jù)持久化、ral調(diào)用相關(guān)代碼。

其下有三個目錄:

|-- infrastructure | |-- event // 領(lǐng)域事件實現(xiàn) | | |-- init.go | | `-- vipcode | | `-- consume_order.go | |-- persistence // 持久化存儲實現(xiàn) | | |-- init.go | | |-- lawyer | | | |-- po.go | | | |-- repo.go | | | `-- transform.go | | `-- vipcode | | |-- po.go | | |-- repo.go | | `-- transform.go | `-- rpc // rpc調(diào)用實現(xiàn) | |-- db | |-- init.go | |-- redis

event:領(lǐng)域事件具體實現(xiàn),依賴rpc服務(wù)。

persistence:放置儲存持久化代碼,數(shù)據(jù)庫存儲對應(yīng)PO,和PO到實體的轉(zhuǎn)化方法,所有具體實現(xiàn)方法都出參都需要轉(zhuǎn)化成實體供給上層使用。

rpc:rpc遠程調(diào)用其他微服務(wù)、消息中間件等服務(wù)代碼。

GEEK TALK

04

總結(jié)

用好DDD的關(guān)鍵,就是理解DDD和核心思想,其本質(zhì)也是面向?qū)ο蟮脑O(shè)計方法,即是把業(yè)務(wù)模型轉(zhuǎn)換為對象模型從而來控制業(yè)務(wù)持續(xù)變化而導(dǎo)致系統(tǒng)的復(fù)雜性,使得系統(tǒng)更加具有可擴展性、可維護性。

在相對比較小、邏輯簡單的微服務(wù),在代碼實現(xiàn)層面,我們并沒有按照DDD進行開發(fā),傳統(tǒng)的MVC足以應(yīng)對,若強行使用DDD則會徒增大家的工作量。

DDD的核心是通過戰(zhàn)略設(shè)計來匹配產(chǎn)品層面的業(yè)務(wù)規(guī)劃,在戰(zhàn)術(shù)設(shè)計層面通過對每個模塊進行抽象、建模來完成業(yè)務(wù)梳理劃分邊界,在代碼實現(xiàn)層面來完成設(shè)計文檔到代碼的映射,做到設(shè)計即代碼、代碼即設(shè)計。

而DDD只適用于大型的、復(fù)雜的業(yè)務(wù)場景。切勿為了DDD而DDD。

參考資料:

[1] 《領(lǐng)域驅(qū)動設(shè)計》

[2]《實現(xiàn)領(lǐng)域驅(qū)動設(shè)計》

[3]https://mp.weixin.qq.com/s/y57l-PhzibAjjL3EzPqSow

[4]https://mp.weixin.qq.com/s/_ggIPOvB-ptBanbqqKULxQ

[5]https://mp.weixin.qq.com/s/jU0awhez7QzN_nKrm4BNwg

作者:劉嘿嘿、離夏、立羽

出處:https://mp.weixin.qq.com/s/oXf84xykPVqtjToWrldM9Q

分享到:
標簽:編程 DDD
用戶無頭像

網(wǎng)友整理

注冊時間:

網(wǎng)站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨大挑戰(zhàn)2018-06-03

數(shù)獨一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學(xué)四六

運動步數(shù)有氧達人2018-06-03

記錄運動步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績評定2018-06-03

通用課目體育訓(xùn)練成績評定