Value Object的六项特征:描述性、不变性、概念整体性、替换性、相等性、无副作用。
特征1: 它度量或描述了领域中的某项概念
特征2: 不变性(Immutability)
一个Value Object在创建后就不能再改变了,不过作为某个物件上的描述性属性,他可以被替换掉。house.color = new Color('Red')
(特征4)
特征3: 将相关属性组成一个「概念整体(Conceptual Whole)」
class Product {
name: string;
money: Money;
}
class Money {
amount: number;
currency: string;
}
特征4: 当度量与描述改变时,可以用另一个Value Object 替换
const p = new Product('A', new Money(100, 'USD'));
p.changePrice(new Money(200, 'USD'));
特征5: 相等性
既然Value Object 没有身份标识的概念,那么只要两个Value Object 身上的属性的值都相等,那我们就会说这两个Value Object 是相等的。
特征6: 无副作用
sub(val: number): void {
if (val > this.amount) {
throw new Error('Money amount cannot be subtract to negative');
}
// 回傳新物件!
return new Money(amount - val, this.currency);
}
如何分辨Value Object 与Entity
在做DDD设计时,我们会偏好Value Object更胜Entity。而判断这两者的标准就在于系统在不在乎这个物件的生命周期变化,并非在于有无ID或相关的识别栏位。甚至,同一个概念物件在不同的Bounded Context中,也可能被分别建成Entity与Value Object。
Entity:
Value Object:
DDD是根据业务能力与商业逻辑来设计领域物件,再接着设计资料库的模型,而非反着来。所以一个Entity在资料库的表中通常会有一个栏位放ID,但Value Object不一定不能有ID。
Entity 除了拥有一个ID 以外,还可以包含多个Value Object 或Entity;而Value Object 同时也可以拥有Value Object 与Entity (尽量避免)。