什么才是正确的领域驱动实现架构?

作为一种系统建模方法,DDD同样涉及系统的体系架构设计。区别于分布式、事件驱动、消息总线等架构设计方法,DDD中的架构设计关注前面各章所介绍的聚合、实体、值对象、领域事件、应用服务以及资源库之间的交互方式和风格,并在设计思想上有其独特的考虑。本节内容将针对DDD特有的架构模式展开讨论,包括经典分层架构、整洁架构以及六边形架构。

DDD经典分层架构

在软件开发过程中,分层架构是最常见、也是最基础的一种架构模式。例如,针对一个Web应用程序,我们可以梳理如下图所示的架构图。


上图展示的就是经典的三层架构,包括用户界面层、业务逻辑层和数据访问层。最终,系统操作数据库完成了业务数据的持久化。本节内容将在经典三层架构的基础上,详细分析DDD中的分层架构模式。

  1. 错误的DDD分层架构

在上图的基础上,原则上我们可以设计四层架构、五层架构等多种多层架构体系。每一层次之间通过接口的方式进行交互,可以严格限制跨层调用,也可以支持部分功能的跨层交互以提供分层的灵活性。下图所示的就是在通用的分层架构基础上所构建的DDD经典分层架构。


暂且不论上图中所展示的分层交互是否合理,我们先来讨论图中所展示的分层组件。本质上,分层架构用于处理组件之间的依赖关系,上图展示了DDD中所包含的4种核心组件,即:

  1. 领域层组件

代表整个DDD应用程序的核心,包含聚合、实体、值对象、领域事件、应用服务、资源库等组件。

  1. 基础设施层组件

这里的基础设施组件范围比较广泛,即可以包括通用的工具类服务,也可以包括数据持久化等具体的技术实现方式。领域层组件中的部分抽象接口(如资源库接口)需要通过基础设施提供的服务得以实现,所以基础设施层组件对领域层组件存在依赖关系。

  1. 应用层组件

应用层组件面向用户接口,是系统对领域模型组件的一种简单封装,通常作为一种门面或网关对外提供统一访问入口,在用户接口和领域模型之间起到衔接作用。同时,因为基础设施组件是对领域模型组件部分抽象接口的具体实现,所以应用组件也会使用基础设施组件来完成业务操作。

  1. 用户接口层组件

用户接口处于系统的顶层,直接面向前端应用,调用应用层组件提供的入口完成用户操作。

基于以上关于DDD中技术组件及其依赖关系的分析,我们明确了上图展示的DDD经典分层架构图实际上存在一定的问题,最主要的问题就是领域层对基础设施层存在依赖,这是不合理的。因为领域层中的资源库接口需要借助于具体的数据访问组件才能得到实现,而数据访问组件属于基础设施层组件,所以是基础设施层依赖于领域层,而不是反其道而行。由此,我们也可以得出一个结论,即设计架构分层的前提是明确系统的核心组件,分层体现的就是对这些核心组件的层次和调用关系的梳理。

  1. 正确的DDD分层架构

那么,我们应该如何正确设计DDD的分层架构呢?为了回答这个问题,我们首先需要梳理架构分层的两个核心问题,即:

  1. 领域模型组件作为核心组件和其他组件之间的依赖关系是怎么样的?
  2. 领域模型组件的抽象接口由谁去实现?

这两个问题的答案决定了架构分层的不同表现风格。而为了更好的回答这两个问题,我们需要引入架构设计过程中的一组设计原则,包括:

(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应用程序可以分为四层,即:

  1. 实体层

实体(Entities)层封装业务规则。请注意它们封装了企业级的、最通用的规则,并且当外部环境发生变化时,这些实体是最稳定的。

  1. 用例层

用例(Use Cases)层则包含了具体的应用逻辑,它实现了所有的用户用例。这些用例使得内层的实体能够依靠实体内定义的业务规则来完成系统的用户需求。

  1. 接口适配器层

接口适配(Interface Adapters)层的目的就是进行数据的转换,将面向用户用例和实体层操作的数据结构转换成为面向数据库、消息通信等外部系统所能接收的数据模型。

  1. 框架与驱动器层

框架和驱动(Frameworks&Drivers)层由各种技术实现工具所组成,常见的包括数据库、Web框架、消息中间件等。我们把这些组件放在整个应用程序的最外层,它们对整个系统的架构不造成任何影响。

基于这种分层方式,整洁架构的整体结构如下图所示。


整洁架构的特性非常明确。层次越靠内的组件依赖的内容越少,位于核心的实体层没有任何依赖。层次越靠内的组件与业务的关系越紧密,因而越不可能形成通用的组件。实体层封装了企业级的业务规则,准确地讲,它应该是一个面向业务的领域模型。而用例层是打通内部业务与外部资源的通道,提供了输出端口与输入端口,但它对外展现其实是应用逻辑,或者说是一个用例。在接口适配层中,我们可以进入网关(Gateway)、控制器(Controller)与表示器(Presenter)等具体的适配器组件,用于打通应用业务逻辑与外层的框架和驱动器,从而实现各种用于访问外部资源的适配机制。

DDD六边形架构

DDD分层架构实际上是一种松散分层架构,位于流程上游的用户接口层和应用层,以及位于流程下游的具备数据访问功能的基础设施层都依赖于领域层,事实上已不存在严格意义上的分层概念。领域驱动设计思想认为应该推平分层架构,不使用严格的分层架构来构建系统,六边形架构(Hexagonal Architecture)也就应运而生。六边形架构促使我们转换视角重新审视一个系统。

六边形架构允许一个应用由用户、程序、自动化测试或批处理脚本驱动,并实现与数据库等外部媒介之间的隔离开发和验证。从设计初衷来讲,六角形架构允许隔离应用程序的核心业务并自动测试其行为,这是该架构在DDD领域中得到应用的核心原因。六边形架构的结构如下图所示,该图来自于软件工程大师Vaughn Vernon。


六边形架构同样表现为是一种分层架构,而且也是三层架构,包括应用程序层、领域层和基础设施层,它们之间的依赖关系如下图所示。


位于上图最上面的是应用程序层,这是DDD应用程序与用户或外部程序之间的交互层,通常包含一些系统交互类的代码,例如用户界面、REST API等。领域层位于上图中的中间位置,隔离应用程序和基础设施,包含所有关注和实现业务逻辑的代码。位于上图最下面的是基础设施层,它包含必要的基础结构类组件,例如与数据库交互的代码或者与其他应用程序的REST API调用代码。

就依赖关系而言,上图所示三层架构中的领域层最为稳定和抽象,所以被应用程序和基础设施层所依赖,而应用程序和基础设施层之间不应该存在任何依赖关系。这样做的好处是把应用程序、业务逻辑和基础架构的关注点分离开发,确保每层组件的约束对其他各层组件的影响较小。

最后,我们来讨论组件边界。在六边形架构中,我们通过引入适配器(Adapter)组件实现与数据库、文件系统、应用程序以及其他各种外部组件之间的集成。

如果你采用的是六边形架构,那么系统应该由内而外围绕领域组件展开,而划分系统的内外部组件成为架构搭建的切入点。可以看到,领域组件位于六边形架构的最内层,应用程序也可以包含业务逻辑,与领域组件构成系统的内部基础架构。而对于外部组件而言,通过各种适配器实现数据持久化、消息通信、各种上下文集成以及用户交互。基于依赖注入和Mock机制,我们可以方便地对适配器组件进行模拟和替换。

DDD架构的映射性

讲到这里,你可能会问,DDD所具备的经典分层架构、整洁架构、六边形架构等多种架构模式之间是否存在一些共性呢?答案是肯定的。事实上,通过分析,我们会发现这些架构的底层逻辑是高度一致性。

我们先来看分层架构和整洁架构,这两种架构模式的对应关系如下图所示。


通过上图的表现形式,我们不难看出,整洁架构和分层架构本质上是一致的,不同的只是具体的分层方式而已。

如果我们把讨论范围扩大到六边形架构,那么可以得到如下图所示的架构映射图。该图清晰展示了不同架构模式所具备的相通性。


我们来对上图进行分析。在上图中,我们可以看到把三种架构模式的组成结构分成了两个部分,即内核层(红色框所包含的部分)和外部层(红色框外部部分)。其中,内核层由应用层和领域层组成,这点对于三种架构模式都是一致的。而对于外部层而言,分层架构和整洁架构包含用户接口和基础设施,而六边形架构则由一系列适配器所组成。基于前面内容的分析,六边形架构中的适配器具备数据持久化、消息通信、各种上下文集成以及用户交互能力,相当于充当了用户接口以及基础设施的功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/606018.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

创建和管理数据库

1. 一条数据的存储过程 存储数据是处理数据的第一步.只有正确的把数据存储起来,我们才能进行有效的处理和分析.否则,只能是一团乱麻.在MySQL中,一个完整的数据存储过程一共有四步 : 创建数据库,确认字段,创建数据表&a…

[图解]SysML和EA建模住宅安全系统-01

1 00:00:00,980 --> 00:00:03,100 接下来,我们来看一下案例 2 00:00:04,930 --> 00:00:06,750 我们这次课程的案例 3 00:00:07,090 --> 00:00:13,800 选用了SysML实用指南的书上 4 00:00:13,810 --> 00:00:16,180 第十七章这个案例 5 00:00:16,350 …

RSAC 2024现场:谷歌展望大模型在网络安全领域的前景

人类距离将网络安全的控制权交给生成式AI还有多远? 前情回顾RSAC2024动态 伪造内容鉴别厂商Reality Defender斩获2024 RSAC创新沙盒冠军 RSAC 2024上值得关注的10款网络安全产品 RSAC 2024创新沙盒十强出炉,谁能夺冠? 安全内参5月8日消息…

代码随想录算法训练营第二十一天:树树树

代码随想录算法训练营第二十一天:树树树 ‍ 513.找树左下角的值 力扣题目链接​**(打开新窗口)** 给定一个二叉树,在树的最后一行找到最左边的值。 示例 1: ​​ 示例 2: ​​ #算法公开课 《代码随想录》算法视频公开课…

跨平台美学!使用DevExpress Reports Office File API时如何管理字体?

DevExpress Office File API是一个专为C#, VB.NET 和 ASP.NET等开发人员提供的非可视化.NET库。有了这个库,不用安装Microsoft Office,就可以完全自动处理Excel、Word等文档。开发人员使用一个非常易于操作的API就可以生成XLS, XLSx, DOC, DOCx, RTF, CS…

鲁大师4月电动两轮车榜:RideyFUN Air智驾系统上线,九号F2z 110智能化再升级

鲁大师4月电动两轮车排行榜数据来源于鲁大师智慧实验室,测评的车型均为市面上主流品牌的主流车型。截止目前,鲁大师智能化电动车测评的车型高达160余台,且还在不断增加和丰富中。 鲁大师电动车智能化测评体系包含车辆的状态采集与管理硬件系统…

android基础-通知

基于第一行android 使用PendingIntent来实现点击通知跳转到另一个活动。 notificationcompat.builder后面可以跟很多的方法,不同方法不同效果,比如加一个音频.setSound(uri) 内置网页 解析json

Spring IoCDI(1)—入门

目录 一、IoC & DI入门 1、Spring是什么 (1)什么是容器? (2)什么是IoC? 二、IoC介绍 1、传统程序开发 2、解决方案 3、IoC程序开发 4、IoC优势 三、DI介绍 通过前面的学习,我们知…

开源即时通讯IM框架 MobileIMSDK v6.5 发布

一、更新内容简介 本次更新为次要版本更新,进行了bug修复和优化升级(更新历史详见:码云 Release Notes、Github Release Notes)。 MobileIMSDK 可能是市面上唯一同时支持 UDPTCPWebSocket 三种协议的同类开源IM框架。轻量级、高…

OpenGVLab/InternVL-Chat-V1-5-Int8

openedai-vision 代码仓库 OpenGVLab/InternVL-Chat-V1-5-Int8 模型文件地址 示 算力平台AutoDL df -h /root/autodl-tmp tar -xvf FileName.tartar -xvf 安装 克隆我们的仓库并跳转到相应目录 2. 创建 conda 环境 cond…

解决Vue devtools插件数据变化不会自动刷新

我们使用devtools插件在监测vuex中表单或自定义组件的数据,发现页面数据发生变化后,但是devtools中还是老数据,必须手动点击devtools刷新才能拿到最新的数据。很烦! 解决方案: 打开chrome的设置,向下翻&…

安服仔养成篇——基线检查

基线检查概念理解 先放白嫖来的解释: “安全基线是保持信息系统安全的机密性、完整性、可用性的最小安全控制,是系统的最小安全保证,最基本的安全要求。” 啥意思呢,基线检查其实就是对操作系统、中间件、数据库、网络设备等多…

2024挂耳式耳机怎么选?5款高性价比开放式耳机推荐榜

近年来,开放式耳机受到了越来越多人的关注,特别是对于运动爱好者来说,在运动的过程中,传统的有线耳机不适合户外运动,不仅佩戴不稳,线还容易缠绕,而普通的蓝牙耳机长时间佩戴会感觉耳朵不适。在…

武汉星起航:亚马逊五大促销类型全面解析,打造销售狂欢新篇章

在全球电商领域,亚马逊以其卓越的平台优势和创新的促销策略,为卖家和消费者搭建了一座互通的桥梁。今天,武汉星起航在这里解析亚马逊的五大促销类型,帮助卖家和消费者更好地把握商机,享受购物的乐趣。 一.…

sql 注入 1

当前在email表 security库 查到user表 1、第一步,知道对方goods表有几列(email 2 列 good 三列,查的时候列必须得一样才可以查,所以创建个临时表,select 123 ) 但是你无法知道对方goods表有多少列 用order …

如何在六个月内学会任何一门外语(ted转述)

/仅作学习和参考,勿作他用/ a question : how can you speed up learning? 学得快,减少在学校时间 结果去研究心理学惹 spend less time at school. if you learn really fast , you donot need to go to school at all. school got in the way of …

邮箱Webhook API发送邮件的性能怎么优化?

邮箱Webhook API发送邮件的步骤?如何用邮箱API发信? 随着业务规模的扩大,如何高效地通过邮箱Webhook API发送邮件,成为了许多企业面临的关键问题。下面,AokSend将探讨一些优化邮箱Webhook API发送邮件性能的方法。 邮…

Tomcat端口占用解决方案

Windows操作系统 出现这种情况: Error was Port already in use :40001;nested exception is :java.net.BindException: Address already in use : JVM_Bind; 步骤1:按下winR键,输入cmd 步骤2:输入以下命令 netstat …

基于D1开发板和腾讯云nginx服务器构建家庭视频监控方案

腾讯云服务器使用nginx搭建rtmp服务器 什么是nginx? nginx是一款优秀的反向代理工具,通过nginx可以实现搭建高可用的轻量级web服务器,除此之外,通过Nginx自带的rtmp模块,也可以实现rtmp服务器的搭建。 安装nginx 安装编…

ESD静电问题 | 摄像头空气放电重启

【转自微信公众号:必学大课堂】
最新文章