作为一种系统建模方法,DDD同样涉及系统的体系架构设计。区别于分布式、事件驱动、消息总线等架构设计方法,DDD中的架构设计关注前面各章所介绍的聚合、实体、值对象、领域事件、应用服务以及资源库之间的交互方式和风格,并在设计思想上有其独特的考虑。本节内容将针对DDD特有的架构模式展开讨论,包括经典分层架构、整洁架构以及六边形架构。
DDD经典分层架构
在软件开发过程中,分层架构是最常见、也是最基础的一种架构模式。例如,针对一个Web应用程序,我们可以梳理如下图所示的架构图。
上图展示的就是经典的三层架构,包括用户界面层、业务逻辑层和数据访问层。最终,系统操作数据库完成了业务数据的持久化。本节内容将在经典三层架构的基础上,详细分析DDD中的分层架构模式。
- 错误的DDD分层架构
在上图的基础上,原则上我们可以设计四层架构、五层架构等多种多层架构体系。每一层次之间通过接口的方式进行交互,可以严格限制跨层调用,也可以支持部分功能的跨层交互以提供分层的灵活性。下图所示的就是在通用的分层架构基础上所构建的DDD经典分层架构。
暂且不论上图中所展示的分层交互是否合理,我们先来讨论图中所展示的分层组件。本质上,分层架构用于处理组件之间的依赖关系,上图展示了DDD中所包含的4种核心组件,即:
- 领域层组件
代表整个DDD应用程序的核心,包含聚合、实体、值对象、领域事件、应用服务、资源库等组件。
- 基础设施层组件
这里的基础设施组件范围比较广泛,即可以包括通用的工具类服务,也可以包括数据持久化等具体的技术实现方式。领域层组件中的部分抽象接口(如资源库接口)需要通过基础设施提供的服务得以实现,所以基础设施层组件对领域层组件存在依赖关系。
- 应用层组件
应用层组件面向用户接口,是系统对领域模型组件的一种简单封装,通常作为一种门面或网关对外提供统一访问入口,在用户接口和领域模型之间起到衔接作用。同时,因为基础设施组件是对领域模型组件部分抽象接口的具体实现,所以应用组件也会使用基础设施组件来完成业务操作。
- 用户接口层组件
用户接口处于系统的顶层,直接面向前端应用,调用应用层组件提供的入口完成用户操作。
基于以上关于DDD中技术组件及其依赖关系的分析,我们明确了上图展示的DDD经典分层架构图实际上存在一定的问题,最主要的问题就是领域层对基础设施层存在依赖,这是不合理的。因为领域层中的资源库接口需要借助于具体的数据访问组件才能得到实现,而数据访问组件属于基础设施层组件,所以是基础设施层依赖于领域层,而不是反其道而行。由此,我们也可以得出一个结论,即设计架构分层的前提是明确系统的核心组件,分层体现的就是对这些核心组件的层次和调用关系的梳理。
- 正确的DDD分层架构
那么,我们应该如何正确设计DDD的分层架构呢?为了回答这个问题,我们首先需要梳理架构分层的两个核心问题,即:
- 领域模型组件作为核心组件和其他组件之间的依赖关系是怎么样的?
- 领域模型组件的抽象接口由谁去实现?
这两个问题的答案决定了架构分层的不同表现风格。而为了更好的回答这两个问题,我们需要引入架构设计过程中的一组设计原则,包括:
(1)依赖性和稳定性原则
组件设计包含一系列原则,其中有三条原则与分层有直接的关系,分别是无环依赖原则、稳定依赖原则和稳定抽象原则。
无环依赖原则(Acyclic Dependencies Principle,ADP)指的是在组件的依赖关系中不能出现环路。稳定依赖原则(Stable Dependencies Principle,SDP)认为被依赖者应该比依赖者更稳定,也就是说如果组件B不如组件A稳定的话,就不应该让组件A依赖组件B。稳定抽象原则(Stable Abstractions Principle,SAP)强调组件的抽象程度应该与其稳定程度保持一致。稳定与抽象是相辅相成的,两者之间的关系示意可以参考下图。
下图 稳定与抽象的关系示例
在上图中,组件X是一个稳定且抽象的组件,因为它被多个组件所依赖。而组件Y则是不稳定的,意味着它也不可能很抽象。那么针对位于同一层级的组件A、B和C而言,它们的抽象和稳定性又应该如何把控呢?我们可以使用单一抽象层次原则(Single Level of Abstraction Principle,SLAP)。良好的分层架构要求一个方法中的所有操作都处于相同的抽象层次,即遵循所谓的单一抽象层次原则。
(2)依赖倒置原则
领域模型组件作为系统的核心理应是抽象且稳定的,也就是说它应该位于系统分层的底端,从而能被其他组件所依赖。用户接口组件直接面向用户,通常是最不稳定的,自然处于系统的顶层。而应用组件处于用户接口组件和领域模型组件之间,这点同样没有异议。那么剩下的就是需要明确基础设施组件的定位,也就是回答领域模型组件的抽象接口由谁去实现这一问题,这就需要进一步引入依赖倒置原则。
依赖倒置原则(Dependency Inversion Principle,DIP)也认为,高层组件不应该依赖于底层组件,两者都应该依赖于抽象;抽象不应该依赖于细节,细节应该依赖于抽象。
我们明确,各种具体的实现技术都不应该包含在领域模型组件中。以数据持久化技术为例,通常我们会以接口的方式抽象数据访问操作,然后通过依赖注入把实现这些数据访问接口的组件注入到领域模型中。这些数据访问的具体实现就可以统一放在基础设施组件中,也就是说基础设施组件实现了领域模型组件中的抽象接口。
基于以上分析,我们可以梳理各个层次之间的关系,从而形成正确的DDD分层架构,如下图所示。
上图的表现形式符合前面各条架构设计原则中的描述,我们通过分层架构管理了DDD中的组件依赖关系。
DDD整洁架构
本质上,所谓的整洁架构也是对DDD四大技术组件进行合理分层的一种架构模式。在架构设计时,整洁架构指导开发人员设计出干净的应用层和领域模型层,确保它们对业务逻辑的专注度,而不掺杂任何具体的技术实现,从而完成领域模型与技术实现之间的完全隔离。
在整洁架构中,一个DDD应用程序可以分为四层,即:
- 实体层
实体(Entities)层封装业务规则。请注意它们封装了企业级的、最通用的规则,并且当外部环境发生变化时,这些实体是最稳定的。
- 用例层
用例(Use Cases)层则包含了具体的应用逻辑,它实现了所有的用户用例。这些用例使得内层的实体能够依靠实体内定义的业务规则来完成系统的用户需求。
- 接口适配器层
接口适配(Interface Adapters)层的目的就是进行数据的转换,将面向用户用例和实体层操作的数据结构转换成为面向数据库、消息通信等外部系统所能接收的数据模型。
- 框架与驱动器层
框架和驱动(Frameworks&Drivers)层由各种技术实现工具所组成,常见的包括数据库、Web框架、消息中间件等。我们把这些组件放在整个应用程序的最外层,它们对整个系统的架构不造成任何影响。
基于这种分层方式,整洁架构的整体结构如下图所示。
整洁架构的特性非常明确。层次越靠内的组件依赖的内容越少,位于核心的实体层没有任何依赖。层次越靠内的组件与业务的关系越紧密,因而越不可能形成通用的组件。实体层封装了企业级的业务规则,准确地讲,它应该是一个面向业务的领域模型。而用例层是打通内部业务与外部资源的通道,提供了输出端口与输入端口,但它对外展现其实是应用逻辑,或者说是一个用例。在接口适配层中,我们可以进入网关(Gateway)、控制器(Controller)与表示器(Presenter)等具体的适配器组件,用于打通应用业务逻辑与外层的框架和驱动器,从而实现各种用于访问外部资源的适配机制。
DDD六边形架构
DDD分层架构实际上是一种松散分层架构,位于流程上游的用户接口层和应用层,以及位于流程下游的具备数据访问功能的基础设施层都依赖于领域层,事实上已不存在严格意义上的分层概念。领域驱动设计思想认为应该推平分层架构,不使用严格的分层架构来构建系统,六边形架构(Hexagonal Architecture)也就应运而生。六边形架构促使我们转换视角重新审视一个系统。
六边形架构允许一个应用由用户、程序、自动化测试或批处理脚本驱动,并实现与数据库等外部媒介之间的隔离开发和验证。从设计初衷来讲,六角形架构允许隔离应用程序的核心业务并自动测试其行为,这是该架构在DDD领域中得到应用的核心原因。六边形架构的结构如下图所示,该图来自于软件工程大师Vaughn Vernon。
六边形架构同样表现为是一种分层架构,而且也是三层架构,包括应用程序层、领域层和基础设施层,它们之间的依赖关系如下图所示。
位于上图最上面的是应用程序层,这是DDD应用程序与用户或外部程序之间的交互层,通常包含一些系统交互类的代码,例如用户界面、REST API等。领域层位于上图中的中间位置,隔离应用程序和基础设施,包含所有关注和实现业务逻辑的代码。位于上图最下面的是基础设施层,它包含必要的基础结构类组件,例如与数据库交互的代码或者与其他应用程序的REST API调用代码。
就依赖关系而言,上图所示三层架构中的领域层最为稳定和抽象,所以被应用程序和基础设施层所依赖,而应用程序和基础设施层之间不应该存在任何依赖关系。这样做的好处是把应用程序、业务逻辑和基础架构的关注点分离开发,确保每层组件的约束对其他各层组件的影响较小。
最后,我们来讨论组件边界。在六边形架构中,我们通过引入适配器(Adapter)组件实现与数据库、文件系统、应用程序以及其他各种外部组件之间的集成。
如果你采用的是六边形架构,那么系统应该由内而外围绕领域组件展开,而划分系统的内外部组件成为架构搭建的切入点。可以看到,领域组件位于六边形架构的最内层,应用程序也可以包含业务逻辑,与领域组件构成系统的内部基础架构。而对于外部组件而言,通过各种适配器实现数据持久化、消息通信、各种上下文集成以及用户交互。基于依赖注入和Mock机制,我们可以方便地对适配器组件进行模拟和替换。
DDD架构的映射性
讲到这里,你可能会问,DDD所具备的经典分层架构、整洁架构、六边形架构等多种架构模式之间是否存在一些共性呢?答案是肯定的。事实上,通过分析,我们会发现这些架构的底层逻辑是高度一致性。
我们先来看分层架构和整洁架构,这两种架构模式的对应关系如下图所示。
通过上图的表现形式,我们不难看出,整洁架构和分层架构本质上是一致的,不同的只是具体的分层方式而已。
如果我们把讨论范围扩大到六边形架构,那么可以得到如下图所示的架构映射图。该图清晰展示了不同架构模式所具备的相通性。
我们来对上图进行分析。在上图中,我们可以看到把三种架构模式的组成结构分成了两个部分,即内核层(红色框所包含的部分)和外部层(红色框外部部分)。其中,内核层由应用层和领域层组成,这点对于三种架构模式都是一致的。而对于外部层而言,分层架构和整洁架构包含用户接口和基础设施,而六边形架构则由一系列适配器所组成。基于前面内容的分析,六边形架构中的适配器具备数据持久化、消息通信、各种上下文集成以及用户交互能力,相当于充当了用户接口以及基础设施的功能。