本篇文章主要介绍了聚合根,聚合的概念,然后介绍了聚合的设计过程和原则,以及对比了聚合,聚合根,实体,值对象的特点。
思考的问题
概念和职责
聚合根:
职责:
-
作为实体,具备自己的业务属性,业务行为,业务逻辑
-
作为聚合的管理者, 在聚合内部,负责协调实体和值对象按照固定的业务规则协同完成共同的业务逻辑;
-
在聚合之间:它是聚合对外的接口人,以聚合根ID的方式接受外部请求和任务,实现上下文中的聚合之间的业务协同;
-
聚合之间通过聚合根关联引用,如果需要访问其他聚合的实体,先访问聚合根,再导航到聚合内部的实体;即外部对象不能直接访问聚合内的实体;
解决的问题:
数据处理方式 |
问题 |
传统:实体与数据模型一一对应, |
任由实体无控制的修改数据,容易导致实体之间数据逻辑不一致的问题;加锁增加了软件复杂度和降低系统性能; |
聚合
在DDD中,实体和值对象都是很基础的领域对象。
聚合是什么呢?类比一下:
DDD的概念 |
人类 |
实体,值对象 |
人 |
聚合 |
社团,组织,部门 |
聚合的好处: |
|
让实体和值对象协同工作的组织就是聚合,用来确保这些领域对象在实现公共的业务逻辑的时候,可以保持数据的一致性 |
个人是组织的一员,协同工作,有共同目标,可以发挥出更大的力量 |
聚合的另一种视图:
-
聚合是业务和逻辑紧密关联的实体和值对象组合而成,聚合是数据修改和持久化的基本单元,一个聚合对应一个数据的持久化;
-
聚合在DDD分层架构中属于领域层,领域层包含了多个聚合,共同实现核心业务逻辑,聚合内的实体以充血模型实现个体业务能力,以及业务逻辑的高内聚;
-
跨多个实体的业务逻辑通过领域服务来实现,跨多个聚合的业务逻辑通过应用服务来实现;
跨越场景 |
处理办法 |
业务场景需要一个聚合中的A实体和B实体共同完成 |
业务逻辑用领域服务来实现; |
业务逻辑需要聚合C和聚合D共同完成 |
应用服务来组合这两个服务; |
聚合的组成:
组成部分 |
说明 |
上下文边界 |
边界是根据业务单一职责和高内聚原则,定义了聚合内部应该包含哪些实体和值对象 |
聚合之间的边界是松耦合的; |
|
聚合根 |
见下面的介绍 |
聚合的设计过程和原则
如何设计聚合?
例子:保单系统聚合的过程。
步骤 |
说明 |
1 |
通过事件风暴的用例分析,场景分析,用户旅程分析得到领域对象(账户,客户,投保单,报废,标的,报价规则,地址,投保人,被保人) |
2 |
从领域对象中找出适合作为实体管理者的对象(聚合根),[是否具备唯一标识,能管理其他的实体]-》(投保,客户) |
3 |
按照业务的职责单一和高内聚原则,找出聚合根紧密关联的实体和值对象,构建以聚合根为中心的集合,即得到聚合;-》(投保聚合,客户聚合) |
4 |
在聚合内部画出聚合根跟实体和值对象的引用和依赖关系图-》(客户依赖地址和账户) |
5 |
多个聚合根据业务语义,和上下文划在同一个限界上下文中 |
聚合的设计原则:
原则 |
说明 |
聚合设计的尽量小 |
如果聚合设计的过大,内部还有大量的实体和值对象,管理会比较复杂,高频操作会有并发和数据库锁冲突的问题,导致系统可用性降低;聚合设计的足够小,也就降低了复杂度,可复用性也更高,降低了后期重构复杂聚合的成本; |
聚合应该高内聚 |
封装的是真正的不变的领域对象,内部的实体和值对象按照固定的规则运行,实现数据的一致性,边界外的任何东西都于该聚合无关, |
通过唯一标识符引用其它聚合 |
聚合之间通过聚合根的唯一ID来关联,而不是直接对象引用的方式,外部的聚合对象如果在本聚合范围内管理,容易导致边界不清晰,增加聚合之间的耦合度; |
边界之外使用最终一致性 |
聚合内部数据强一致性,聚合之间数据最终一致性,在一次事务中最多只修改一个聚合的数据状态,如果在一次事务中涉及修改多个聚合的状态,应该使用领域事件的方式来异步的实现最终一致性,实现聚合之间的解耦; |
在应用层实现跨聚合的调用 |
实现微服务内部聚合之间的解耦,以为未来以聚合为单位的拆分和组合,应该避免跨聚合的的领域服务调用和数据表关联; |
案例
对象之间是否必须保持一些固定的规则
比如:
-
Order(一个订单)必须有对应的客户邮寄信息,否则就不能称为一个有效的Order;
-
同理,Order对OrderLineItem有不变性约束,Order也必须至少有一个OrderLineItem(一条订单明细),否则就不能称为一个有效的Order;
-
另外,Order中的任何OrderLineItem的数量都不能为0,否则认为该OrderLineItem是无效的,同时可以推理出Order也可能是无效的。因为如果允许一个OrderLineItem的数量为0的话,就意味着可能会出现有OrderLineItem的数量都为0,这就导致整个Order的总价为0,这是没有任何意义的,是不允许的,从而导致Order无效;所以,必须要求Order中所有的OrderLineItem的数量都不能为0;
-
那么现在可以确定的是Order必须包含一些OrderLineItem,那么应该是通过引用的方式还是ID关联的方式来表达这种包含关系呢?
-
这就需要引出另外一个问题,那就是先要分析出是OrderLineItem是否是一个独立的聚合根。回答了这个问题,那么根据上面的规则就知道应该用对象引用还是用ID关联了。
-
那么OrderLineItem是否是一个独立的聚合根呢?因为聚合根意味着是某个聚合的根,而聚合有代表着某个上下文边界,而一个上下文边界又代表着某个独立的业务场景,这个业务场景操作的唯一对象总是该上下文边界内的聚合根。想到这里,我们就可以想想,有没有什么场景是会绕开订单直接对某个订单明细进行操作的。也就是在这种情况下,我们是以OrderLineItem为主体,完全是在面向OrderLineItem在做业务操作。有这种业务场景吗?没有,我们对OrderLineItem的所有的操作都是以Order为出发点,我们总是会面向整个Order在做业务操作,比如向Order中增加明细,修改Order的某个明细对应的商品的购买数量,从Order中移除某个明细,等等类似操作,我们从来不会从OrderlineItem为出发点去执行一些业务操作;另外,从生命周期的角度去理解,那么OrderLineItem离开Order没有任何存在的意义,也就是说OrderLineItem的生命周期是从属于Order的。所以,我们可以很确信的回答,OrderLineItem是一个实体。
总结
聚合的特点:
聚合根的特点:
-
聚合根是实体,具备唯一标识,有独立的生命周期,一个聚合只有一个聚合根,聚合根在聚合之内采用引用依赖的方式对实体和值对象进行组织和协调,聚合根和聚合根之间通过唯一id进行聚合之间的协同;
-
实体的特点:具备id标识,可以通过id进行相等性比较,实体在聚合内唯一,但是状态可变,它依附于聚合根,它的生命周期由聚合根管理,实体一般都会持久化,跟数据持久化对象存在多种对应关系(一对一,一对多,多对一,1对0),实体可以引用聚合中的聚合根,实体,值对象;
-
值对象特点:无id,不可变,无生命周期,用完即失效,值对象之间通过属性值判断相等性,他的核心是值,是一组概念完整的属性集合,用于描述实体的特征和状态,值对象尽量只引用值对象;
下面我们来回答最初的两个问题:
为什么要在实体和限界上下文之间增加聚合和聚合根,作用是什么?
如何设计聚合?
-
过程是:
-
通过事件风暴(用例分析,场景分析,用户旅程分析)得到实体和值对象
-
然后找出聚合根,按照高内聚低耦合的设计原则,找出跟聚合根紧密关联的实体和值对象,即形成聚合
-
并画出聚合内的实体和值对象的引用依赖关系
-
最后把业务把关联紧密的聚合画在同一个限界上线文中,即完成了领域建模
----------------------------------
大家好,我是流水,一个资深的IT从业人员和架构师. 非常高兴您能搜索到,并看到这篇文章,希望这篇文章的内容能给您带来新的知识和帮助。
也欢迎扫描以下的二维码或微信搜索 “superxtech”,关注我的微信公众号 , 我会把更多更好的IT领域技术知识带给您!
----------------------------------