日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

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

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

背景:

 

充血模型是 DDD 分層架構(gòu)中實體設(shè)計的一種方案,可以使關(guān)注點聚焦于業(yè)務(wù)實現(xiàn),可有效提升開發(fā)效率、提升可維護性;

1、DDD 項目落地整體調(diào)用關(guān)系

調(diào)用關(guān)系圖中的 Entity 為實體,從進入領(lǐng)域服務(wù)(Domin)時開始使用,直到最后返回。

2、實體設(shè)計

充血模型是實體設(shè)計的一種方法,簡單來說,就是一種帶有具體行為方法和聚合關(guān)聯(lián)關(guān)系的特殊實體;

關(guān)于實體設(shè)計,需要明白的關(guān)鍵詞為:領(lǐng)域服務(wù) -> 聚合 -> 聚合根 -> 實體 -> 貧血模型 -> 充血模型

聚合與聚合根:

聚合是一種關(guān)聯(lián)關(guān)系,而聚合根就是這個關(guān)系成立的基礎(chǔ),沒有聚合根,這個聚合關(guān)系就無法成立;

舉個例子,存在 3 個實體:用戶、用戶組、用戶組關(guān)聯(lián)關(guān)系,這 3 個實體形成的關(guān)聯(lián)關(guān)系就是聚合,而用戶實體就是這個聚合中的聚合根;

實體:

定義在領(lǐng)域?qū)?,是領(lǐng)域?qū)拥闹匾兀瑥念I(lǐng)域劃分到工程實踐落地,都應(yīng)該圍繞實體進行,DDD 中的實體和數(shù)據(jù)庫表不只是 1 對 1 關(guān)系,可能是 1 對多或者僅為內(nèi)存中的對象;

貧血模型:

實體不帶有任何行為方法,也不帶有聚合關(guān)聯(lián)關(guān)系,作用基本相當(dāng)于值對象(ValueObject),僅作為值傳遞的對象,和傳統(tǒng)三層項目架構(gòu)中的實體具有相同作用,不建議使用。補充說明:一般我們使用的 DTO 就可以被當(dāng)做是值對象

充血模型:

實體中帶有具有行為方法和聚合關(guān)聯(lián)關(guān)系,行為方法是說 create、save、delete 等封裝了一類可以指代行為的方法,比如在用戶實體對象中具有用戶組實體的引用,這樣當(dāng)我們需要操作用戶組時,只通過用戶實體進行操作就可以。

工程實踐中,建議采用充血模型,好處是隱藏膠水代碼,提升代碼可讀性,使關(guān)注點聚焦于業(yè)務(wù)實現(xiàn)。

充血模型在實踐中的問題:

行為代碼量過多,導(dǎo)致實體內(nèi)部臃腫膨脹,難以閱讀,難以維護,對于這種問題,我們需要根據(jù)實體行為的代碼量多少來采取不同的解決方案。

解決方案:

場景 1:行為不會導(dǎo)致實體臃腫的情況下,在實體中完成行為定義

public CooperateServicePackageConfig save() {

// 直接調(diào)用基礎(chǔ)設(shè)施層進行保存

cooperateServicePackageConfigRepository.save(this);

return this;

}

場景 2:行為導(dǎo)致實體臃腫的情況下,采用外部定義行為的方式,核心思想是借助其他類實現(xiàn)行為代碼定義,將臃腫代碼外移,保留干凈的實體行為:

1)創(chuàng)建工具類,將某個實體中的行為定義其中,實體負(fù)責(zé)調(diào)用該工具類

public CooperateServicePackageConfig save() {

// 將處理過程放在工具類中

ServicePackageSaveUtils.save(this);

return this;

}

2)創(chuàng)建新實體,將該實體的使用場景明確至某個細分行為,比如一個聚合根(ExampleEntity)的保存可能涉及到 5 個實體的保存,那么我們定義一個 ExampleSaveEntity 實體,專門用來處理該聚合下的保存行為

實踐經(jīng)驗:

1、關(guān)于 spring bean 注入:充血模型在實體中使用靜態(tài)注入方法實現(xiàn)。例:

private LabelInfoRepository labelInfoRepository = ApplicationContextUtils.getBean(LabelInfoRepository.class);

2、充血模型的實體序列化,排除非必要屬性,在一些 redis 對象緩存時可能會用到。例:

// 使用注解排除序列化屬性

@Getter(AccessLevel.NONE)

private LabelInfoRepository labelInfoRepository = ApplicationContextUtils.getBean(LabelInfoRepository.class);

// 使用注解排除序列化屬性

@JSONField(serialize = false)

private ServicePackageConfig servicePackageConfig;

// 使用注解排除序列化 get 方法

@Transient

@JSONField(serialize = false)

public static CooperateServicePackageRepositoryQuery getAllCodeQuery(Long contractId) {

CooperateServicePackageRepositoryQuery repositoryQuery = new CooperateServicePackageRepositoryQuery();

repositoryQuery.setContractIds(com.google.common.collect.Lists.newArrayList(contractId));

repositoryQuery.setCode(RightsPlatformConstants.CODE_ALL);

return repositoryQuery;

}

3、利用 Set 方法建立聚合綁定關(guān)系。例:

public void setServiceSkuInfos(List<ServiceSkuInfo> serviceSkuInfos) {

if (CollectionUtils.isEmpty(serviceSkuInfos))

{

return;

}

this.serviceSkuInfos = serviceSkuInfos;

List<String> allSkuNoSet = serviceSkuInfos

.stream()

.map(one -> one.getSkuNo())

.collect(Collectors.toList());

String skuJoinStr = Joiner.on(GlobalConstant.SPLIT_CHAR).join(allSkuNoSet);

this.setSkuNoSet(skuJoinStr);}

 

作者:京東健康 張君毅
來源:京東云開發(fā)者社區(qū)

分享到:
標(biāo)簽:項目 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)練成績評定