DDD原理最全詳解(萬字圖文總結(jié))
DDD
DDD,全稱是“Domain-Driven Design”,翻譯過來就是領(lǐng)域驅(qū)動設(shè)計,是一種以領(lǐng)域為中心的軟件開發(fā)方法論。
圖片
DDD,強調(diào)通過深入理解業(yè)務(wù)領(lǐng)域,并將領(lǐng)域知識明確地融入到系統(tǒng)設(shè)計中,從而來構(gòu)建復(fù)雜的軟件系統(tǒng)。
DDD適用于各種復(fù)雜的業(yè)務(wù)系統(tǒng),尤其適用于:業(yè)務(wù)邏輯復(fù)雜、系統(tǒng)規(guī)模龐大...等應(yīng)用場景。
DDD原理
DDD強調(diào)建立清晰的領(lǐng)域模型,使開發(fā)人員、和業(yè)務(wù)人員,能夠以共同的語言進行交流。
DDD,將系統(tǒng)架構(gòu)分為多個層次,如下圖所示:
圖片
主要會包含,如下四層:
領(lǐng)域?qū)樱―omain Layer)
包含業(yè)務(wù)邏輯、和領(lǐng)域模型,聚合了實體、和值對象...等等;
應(yīng)用層(Application Layer)
負責:協(xié)調(diào)應(yīng)用程序的行為,處理用戶請求;
協(xié)調(diào)領(lǐng)域?qū)樱{(diào)用領(lǐng)域服務(wù)來完成用戶請求。
接口層(Interface Layer)
提供用戶、與系統(tǒng)交互的接口,比如:通過HTTP...等協(xié)議,接收用戶請求。
以及,將領(lǐng)域模型轉(zhuǎn)換為DTO(Data Transfer Object),以便于傳輸。
基礎(chǔ)設(shè)施層(Infrastructure Layer)
處理數(shù)據(jù)持久化、和外部服務(wù)交...等等。
這里面,最重要的就是領(lǐng)域?qū)印?/p>
領(lǐng)域模型
領(lǐng)域模型,將業(yè)務(wù)領(lǐng)域抽象成一個模型,這個模型包含了領(lǐng)域中的實體、值對象、聚合、服務(wù)...等。
圖片
實體
具有唯一標識符的對象,代表業(yè)務(wù)中的一個重要概念,比如:用戶、訂單、產(chǎn)品...等。
圖片
public class User {
private String userId; // 唯一標識
private String name;
private String email;
public User(String userId, String name, String email) {
this.userId = userId;
this.name = name;
this.email = email;
}
public String getUserId() {
return userId;
}
public String getName() {
return name;
}
public String getEmail() {
return email;
}
public void updateEmail(String newEmail) {
this.email = newEmail;
}
}
值對象
沒有唯一標識符的對象,通常用來描述實體的特征,比如:地址、或日期...等等。
public class Address {
private String street;
private String city;
private String country;
// ... 其他屬性
// 構(gòu)造方法,確保不可變性
public Address(String street, String city, String country) {
this.street = street;
this.city = city;
this.country = country;
}
// 只提供getter方法,不提供setter方法
}
聚合
由相關(guān)實體、和值對象,組成的集合,是數(shù)據(jù)修改、和持久化的基本單元。
public class Order {
private Long id;
private User customer;
private List<OrderItem> items;
private Address shippingAddress;
// ... 其他屬性和方法
// 構(gòu)造方法,設(shè)置聚合根
public Order(User customer, Address shippingAddress) {
this.customer = customer;
this.shippingAddress = shippingAddress;
this.items = new ArrayList<>();
}
public void addItem(OrderItem item) {
items.add(item);
}
}
- 聚合根:Order是聚合根,負責維護聚合內(nèi)部的一致性。
- 實體:User、OrderItem都是實體。
- 值對象:Address是值對象
限界上下文
在領(lǐng)域驅(qū)動設(shè)計(DDD)中,限界上下文(Bounded Context) 是一個非常重要的概念。
圖片
它定義了一個明確的領(lǐng)域邊界,在這個邊界內(nèi),領(lǐng)域模型有著一致的業(yè)務(wù)語義和規(guī)則。
簡單來說,限界上下文就是將一個復(fù)雜的領(lǐng)域劃分成多個小的、可管理的子域,每個子域都有自己獨立的模型。
領(lǐng)域服務(wù)
封裝一些領(lǐng)域相關(guān)的操作,不屬于任何實體、或值對象。
當一個操作涉及多個聚合時,可以使用領(lǐng)域服務(wù)來協(xié)調(diào)它們。
比如:我們有一個電商系統(tǒng),其中涉及到訂單、產(chǎn)品和庫存...等概念。
當用戶下單時,我們需要檢查庫存是否充足,如果充足則創(chuàng)建訂單,否則拒絕訂單。
這個過程涉及到多個聚合(訂單、產(chǎn)品)之間的交互,關(guān)注領(lǐng)域模型內(nèi)部的業(yè)務(wù)邏輯,是領(lǐng)域?qū)拥暮诵膶崿F(xiàn)。