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

news2024/12/23 15:52:48

作为一种系统建模方法,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.coloradmin.cn/o/1655558.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

揭秘设计师必备神器:情绪板是什么?

每个伟大的设计项目都从一点灵感开始。无论你是在设计网站、应用程序,还是想重新装修房子,情绪板都可以帮助你激发创造力,甚至情绪板也可以决定UI界面是否成功。本文将分享什么是情绪板,为什么需要情绪板,以及如何充分…

Linux下多线程相关概念

thread 1.什么是线程1.1 线程优缺点1.2 线程异常1.3 线程用途 2. 进程和线程区别3. 线程控制3.1 POSIX线程库3.2 pthread_create()3.3 线程ID3.4 线程ID地址空间布局pthread_self() 3.5 线程终止pthread_exit函数pthread_cancle函数 3.6 线程等待3.7 分离线程__thread修饰全局变…

OpenCV Radon变换探测直线(拉东变换)

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 Radon变换可以将原始图像中直线特征的处理问题转化为变换域图像中对应点特征的处理问题,其中对应特征点的横坐标表示原始图像的旋转角度,一般来讲原始图像中的噪声不会分布在直线的特征上。因此,Radon变换在探测…

Python实战开发及案例分析(12)—— 模拟退火算法

模拟退火算法(Simulated Annealing)是一种概率搜索算法,源自于金属退火过程。在金属退火中,通过缓慢降低温度,金属内部的原子能够从高能态逐步达到较低能态。模拟退火算法利用类似的原理,通过随机搜索和概率…

吉时利2400与Keithley 2450 SMU 数字源表区别?

Keithley SMU(源测量单元)数字源表是一种精密的电子测试设备,它结合了电流和电压源以及测量功能。这些设备被设计用于需要紧密耦合源和测量的测试应用中。Keithley 2400系列SMU数字源表提供了四象限精密电压和电流源/负载,以及触摸…

远程智控BACnet/IP I/O模块助力Metasys系统无缝对接

江森自控的Metasys系统以其强大的综合管理能力成为众多楼宇自控项目的首选平台。然而,面对日益增长的个性化需求与复杂多变的设备接入挑战,如何高效、灵活地扩展其I/O控制能力成为关键。在此背景下,BACnet/IP分布式远程I/O模块的出现&#xf…

可视化大屏的应用:电子政务领域的巨大应用价值

可视化大屏在电子政务领域的应用价值主要体现在以下几个方面: 数据监控与分析 可视化大屏可以将政务数据以图表、地图等形式展示在大屏上,帮助政府部门实时监控和分析各项指标和数据变化。例如,可以实时显示人口统计、经济指标、环境监测等…

如何评估大模型音频理解能力-从Gemini说起

Gemini家族包含Ultra、Pro和Nano三种大小的模型是谷歌开发的大型多模态人工智能模型,它在人工智能的多模态领域实现了重大突破,结合了语言、图像、音频和视频的理解能力。 Gemini的性能评估情况如下: Gemini模型的评估的具体指标从文本理解能…

专题六_模拟(1)

目录 1576. 替换所有的问号 解析 题解 495. 提莫攻击 解析 题解 1576. 替换所有的问号 1576. 替换所有的问号 - 力扣(LeetCode) 解析 题解 class Solution { public:string modifyString(string s) {// 40.专题六_模拟_替换所有的问号_Cint n s.…

Qt跨平台开发demo(适用萌新)

最近需要参与一款Qt跨平台的软件开发,在此之前,特把基础信息做学习和梳理,仅供参考。 所使用的技术和版本情况如下: 虚拟机:VMware 16.2.5操作系统:ubuntu-20.04.6-desktop-amd64:Mysql数据库…

在阿里云K8S容器中,部署websocket应用程序的总结

一、背景 有一个websocket应用程序,使用spring boot框架开发,http端口号是6005,提供的是websocket服务,所以它还监听一个8889端口的tcp协议。 现在要把它部署到阿里云的k8s容器里,本文着重描述service层的配置。 因…

不会pdf修改编辑文字怎么办?看完秒懂

不会pdf修改编辑文字怎么办?在日常生活中,PDF文件已成为我们工作、学习不可或缺的一部分。然而,很多人对PDF文件的编辑操作感到困惑,尤其是修改其中的文字。今天,我们就来详细解析一下,不会PDF修改编辑文字…

C++进阶之路:探索访问限定符、封装与this指针的奥秘(类与对象_上篇)

✨✨ 欢迎大家来访Srlua的博文(づ ̄3 ̄)づ╭❤~✨✨ 🌟🌟 欢迎各位亲爱的读者,感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢,在这里我会分享我的知识和经验。&am…

鸿蒙开发之 if/else:条件渲染

ArkTS提供了渲染控制的能力。条件渲染可根据应用的不同状态,使用if、else和else if渲染对应状态下的UI内容。 使用规则 支持if、else和else if语句。if、else if后跟随的条件语句可以使用状态变量。允许在容器组件内使用,通过条件渲染语句构建不同的子…

数据结构--图。

在前面,我们学习了线性表和树,而接下来我们要学习的图相较于他们就更加复杂。 目录 一.图的有关概念 一.图的有关概念 1.定义 图(graph)G由两个集合V和E组成,记为G(VE)。V是顶点的有穷非空集合;E是边的集合,边是V中顶点的无序对…

02-单片机商业项目编程,从零搭建低功耗系统设计

一、本文内容 上一节《01-单片机商业项目编程,从零搭建低功耗系统设计-CSDN博客》已经对事件驱动原理有个基本了解,本节主要就是如何将事件写的更规范,而不是用t_flag这样的标记,写多了可读性也不强;本节结尾总结将提出…

【探索Java编程:从入门到入狱】Day5

🍬 博主介绍👨‍🎓 博主介绍:大家好,我是 hacker-routing ,很高兴认识大家~ ✨主攻领域:【渗透领域】【应急响应】 【Java、PHP】 【VulnHub靶场复现】【面试分析】 🎉点赞➕评论➕收…

《ESP8266通信指南》13-Lua 简单入门(打印数据)

往期 《ESP8266通信指南》12-Lua 固件烧录-CSDN博客 《ESP8266通信指南》11-Lua开发环境配置-CSDN博客 《ESP8266通信指南》10-MQTT通信(Arduino开发)-CSDN博客 《ESP8266通信指南》9-TCP通信(Arudino开发)-CSDN博客 《ESP82…

驱动比例线圈功率放大器

驱动比例线圈功率放大器是一种用于控制比例电磁铁的电流大小实现被控设备的位移,采用高性能的嵌入式32位微处理器作为运算核心,这些微处理器具有高速指令运行能力,电源24VDC驱动,输入指令兼容性强,输出电流大小可调&am…

云打印怎么保护用户的隐私?

随着互联网的发展,在当下的网络环境下,用户的隐私越来越难以保证安全。特别是对于打印业务来说,盗取用户文件、转卖客户信息的内容时有发生。那么我们作为出色的云打印服务商,该如何保证用户的隐私呢?今天就来给大家介…