DDD实战之五:战略设计之上下文映射和系统分层架构(上)

简介: DDD实战之五:战略设计之上下文映射和系统分层架构(上)

在完成了限界上下文的识别(也就是系统“最粗粒度”的模块划分)后,我们需要对这些上下文之间的协作关系进行分析——即“限界上下文关系映射”。也只有在完成上下文关系映射后,我们才能真正的判定自己所做出的“限界上下文识别”是否真的达到了自己想要的“低耦合、高内聚”的目标。因为,通过“限界上下文映射”我们就能够看到:

  • 这些上下文之间有哪些协作关系?
  • 这些关系是强关联还是弱关联?

关于“限界上下文识别”和“限界上下文关系映射”,我认为这是 DDD 战略设计中最重要的部分,甚至可以说:这两个工作将决定了微服务切分是否有效的关键因素!

但是,肯定会有人说:限界上下文不用 DDD,我凭直觉就能识别出来。我的回答是:是的,你貌似可以!但更重要的是限界上下文的关系映射,这将决定做微服务拆分后、这些微服务之间是怎样做到“高内聚、低耦合”的。如果你不用 DDD 战略设计,我可以很负责任的告诉你:你将来的“微服务”之间的调用关系一定会变得无法控制!

在我实际工作中接触的某大型国企 IT 系统中,所谓业务中台上千万行代码,部署在十多个微服务中心,而 80%以上的外围接口调用、或前端界面服务请求,都要从十个以上的微服务中心全部走一遍!这是不是很可怕的灾难?系统的高可用性、业务需求的快速响应还有什么保障可言?

所以说:掌握好 DDD 战略设计,几乎是做好微服务设计必不可少的前提!不懂得 DDD,你做的“微服务拆分”可能还不如不拆分,单体应用可能更适合你!

在本节内容中,除了搞定限界上下文的映射之外,我还将对系统分层架构进行设计,并在最终给出项目组可直接用于开发的代码框架结构。

需要特别说明的是:在写到这篇的过程中,我反复和多位业界大拿请教,大家普遍认为我前面第三篇中业务子域的分类上“核心子域”太多了。我经过再三考虑,将“核心子域”进行了调整,这将影响到本篇关于系统分层结构的设计,您可以跳回到第三篇重新看下“业务子域识别与分类”这一小段。这里再次重申下:您在阅读中发现我的任何不妥之处 ,可随时与我交流,我发现不妥之处一定认真思考改进,感谢您的支持!

01限界上下文映射


1跨上下文用例识别

为了识别限界上下文之间的映射关系,我们需要对跨上下文的业务用例(也叫业务服务),从架构设计的技术视角绘制服务序列图,进而识别它们之间的映射关系。

首先,第一步,我们识别出所有的跨限界上下文的业务用例。在识别某个业务用例是否跨限界上下文时,一定要注意两个基本事实:1)其实大部分业务用例都是从“群买菜”小程序前端界面发起的,前端与服务端的交互,不算跨限界上下文;2)限界上下文提供的是服务端的业务服务,跨限界上下文一定要是服务端逻辑的相互关系。

关于如何识别“服务端的跨限界上下文”业务逻辑,我认为需要逐个分析前面罗列的所有业务用例,从如下两个角度筛选:

初步分析业务用例内部的逻辑,看是否需要多个上下文来承担职责。如果是,则需将该用例纳入分析范围;

分析业务用例图中的被包含的“子用例”,看是否存在上下文包含了被归类到别的上下文的情况。如果是,则需将该用例纳入分析范围;为此,识别结果如下图罗列:

image.png

需要说明的是:有几个用例看起来好像是跨上下文的,其实不是。分别说明如下:

1. “确认购买并付款”、“创建订单预支付”、“完成订单支付”、“补收客户货款”、“退客户货款”,这几个用例其实是订单上下文和支付系统之间的关系。考虑到我们的系统上下文设计,支付系统其实不是我们目标系统的工作范围,故这里不纳入范围。

2. “后台查询店铺商品”、“后台浏览店铺订单列表”、“管理店铺客户信息”、“管理店铺员工”、“开通店铺加盟并设置分成政策”、“添加品牌店铺到加盟列表”,看起来每个用例的名称上都涉及到 2 个限界上下文、并都跟“店铺”上下文发生关系,其实它们只是需要店铺 ID,并没有与店铺上下文发现任何业务交互,故实际上不作为“跨上下文”的用例来对待。

其次,这些业务用例中,从跨上下文协作关系角度来看,其中有些用例画服务序列图是会出现重复的,我们需要识别出来:

1. “创建新店铺”、“编辑店铺信息”,会用到短信验证码验证手机号,故都属于跨店铺、平台集成两个限界上下文的,且交互逻辑一致,故重复。

2. “加商品到购物车”、“选购接龙商品到购物车”,其实都是涉及到订单和商品两个限界上下文的相同交互关系,故重复。

3. “创建接龙活动”、“编辑接龙活动”,这两个业务用例的操作序列,从跨上下文协作关系角度来看也是完全一样的,只是后者需加载原有“接龙”上下文内部已持久化的信息、前者不用。

4. “创建付款订单”是“确认接龙付款”的子用例,故只需要保留“确认接龙付款”即可。

5. “添加品牌店铺到加盟列表”、“从加盟列表删除品牌店铺”,这两个业务用例都是涉及到店铺和加盟两个上下文的关系,并且都是获取店铺相关信息、并将店铺信息传输给加盟上下文处理,故重复。

最终,我们确定下来,被用来绘制服务序列图,进而确定各上下文协作关系的业务用例罗列如下图:

image.png

2基于跨上下文用例映射上下文关系

我们接下来的工作,就是要对这些业务用例,逐个进行技术分析,设计出服务序列图,然后根据服务序列图确定限界上下文的关系映射。

创建新店铺

创建新店铺涉及到手机号短信验证,故需要跨店铺和平台集成两个上下文。服务序列图如下:

image.png

基于该服务序列图识别出“店铺”和“平台集成”上下文关系如图(C 表示服务调用客户端、S 表示被调用服务端,以下同。DDD 标准方法中一般用“上下游关系”、及诸如“开发主机服务 OHS”等方式表达,但我认为那是废话,并没有清晰的表达强弱关联,故不在这里采用):

image.png

初始化店铺默认选项

根据产品 UI 原型的设计方案,新店铺创建后,系统机器人需要基于异步事件,对店铺的相关选项立即进行初始化:

1. 店铺默认门头图片(UI 允许创建人未设置门头图片);

2. 店铺是否开通订单短信提醒(随着业务发展可能会调整,故不放在创建店铺时设定);

3. 店铺的第一个员工(就是创建人自己);

4. 店铺默认的加盟分销政策(随着业务发展可能会调整,故不放在创建店铺时设定);

5. 店铺的首个接龙地点(详见产品 UI 原型,店铺接龙支持多个地点,但首个地点就是店铺地址);

6. 创建商家及商家账户(如果是创建人的第一家店铺);

7. 店铺默认商品分类(系统规则会按季节调整);

8. 店铺默认的商品搜索热词(系统后台会不定期根据某种策略更新);

这些默认选项,涉及到“店铺”、“员工”、“商品”、“接龙”、“商家账户”、“鉴权”共 6 个上下文的协作。服务序列图涉及如下:

image.png

根据该服务序列图,我们得出这 6 个限界上下文的协作关系如下图:

image.png

从上图的上下文依赖关系中,看到店铺对员工、商品、接龙、商家账户 4 个上下文产生了调用关系依赖,这是不合理的。因为从业务角度来说,其实是后 4 者依赖于店铺的存在而存在,而不是反过来。为此,我们做这样的调整:

1. 采用消息发布者/订阅者模式,让后 4 者依赖店铺上下文发布的“店铺已创建”消息;

2. 去掉“接龙”上下文对店铺首个接龙地点初始化的逻辑。本质上,仔细分析产品 UI 界面设计的要去,我们发现这其实是“群买菜小程序”在展示创建接龙活动的前端界面时,需调用“店铺”上下文服务来获取的信息:如果所选择店铺已经有接龙地址,则返回已有的接龙地址列表供选择;如果该店铺尚无首个接龙地址,则返回店铺地址作为默认接龙地址。

经过修改后的服务序列图如下:

image.png

根据该服务序列图,我们修改上下文的协作关系如下图(P 表示消息发布者,S 表示消息订阅者,下同):

image.png

加商品到购物车

客户浏览店铺商品,选中感兴趣的商品到购物车,以便后续的购买结算。考虑到添加商品到购物车中去的“时间”特殊性:我们需要在客户将商品加入到购物车时,为商品创建“快照”,以避免商品信息在后面被编辑修改(比如改了图片或描述、尤其是价格)时,影响到对客户的购买承诺。

为此,该业务用例(服务)就涉及到“订单”和“商品”两个上下文,服务序列图设计如下:

image.png

该序列图展示出订单和商品的上下文关系如图:

image.png

发送订单提醒

“发送订单提醒”需要在订单上下文中发起、“通知”上下文中发送通知,故涉及“订单”和“通知”两个上下文。该用例服务序列图如下:

image.png

考虑到订单消息提醒是没有副作用的、而且也不需要保证必须成功,故采用消息发布者/订阅者模式比较合适。也就是得到如下所示的“订单”和“通知”上下文的映射关系:

image.png

创建接龙活动

与“加商品到购物车”用例类似,商家在创建接龙活动、添加商品到接龙中去时也存在“时间”特殊性,需要为商品创建“快照”,以避免商品在后面被编辑修改(比如改了图片或描述、尤其是价格)时,影响到接龙活动的开展。为此,该业务用例(服务)就涉及到“接龙”和“商品”两个上下文,服务序列图设计如下:

image.png

该序列图展示出接龙和商品的上下文关系如图:

image.png

浏览我的接龙

按照产品 UI 设计文档,接龙只能在店铺下存在。而“浏览我的接龙”实际上是“浏览我被授权操作的所有店铺发布的、或其被授权店铺所加盟品牌店铺发布的、或我参与的”的所有接龙。故该用例涉及到“接龙”、“店铺”两个上下文,服务序列图如下:

image.png

相关文章
|
24天前
|
存储 SQL 网络协议
C语言C/S架构PACS影像归档和通信系统源码 医院PACS系统源码
医院影像科PACS系统,意为影像归档和通信系统。它是应用在医院影像科室的系统,主要的任务是把日常产生的各种医学影像(包括核磁、CT、超声、各种X光机、各种红外仪、显微仪等设备产生的图像)通过各种接口(模拟、DICOM、网络)以数字化的方式海量保存起来,并在需要的时候在一定授权下能够快速地调回使用。同时,PACS系统还增加了一些辅助诊断管理功能。
40 11
|
17天前
|
安全 数据管理 中间件
云LIS系统源码JavaScript+B/S架构MVC+SQLSugar医院版检验科云LIS系统源码 可提供演示
检验科云LIS系统源码是医疗机构信息化发展的重要趋势。通过云计算技术实现数据的集中管理和共享可以提高数据利用效率和安全性;通过高效灵活的系统设计和可扩展性可以满足不同医疗机构的需求;通过移动性和智能化可以提高医疗服务的精准度和效率;通过集成性可以实现医疗服务的协同性和效率。因此,多医院版检验科云LIS系统源码将成为未来医疗机构信息化发展的重要方向之一。
26 2
|
5天前
|
前端开发 Java 关系型数据库
Java医院绩效考核系统源码B/S架构+springboot三级公立医院绩效考核系统源码 医院综合绩效核算系统源码
作为医院用综合绩效核算系统,系统需要和his系统进行对接,按照设定周期,从his系统获取医院科室和医生、护士、其他人员工作量,对没有录入信息化系统的工作量,绩效考核系统设有手工录入功能(可以批量导入),对获取的数据系统按照设定的公式进行汇算,且设置审核机制,可以退回修正,系统功能强大,完全模拟医院实际绩效核算过程,且每步核算都可以进行调整和参数设置,能适应医院多种绩效核算方式。
26 2
|
14天前
|
API 开发者 UED
构建高效微服务架构:后端开发的新趋势移动应用与系统:开发与优化的艺术
【4月更文挑战第30天】 随着现代软件系统对可伸缩性、灵活性和敏捷性的日益需求,传统的单体应用架构正逐渐向微服务架构转变。本文将探讨微服务架构的核心概念,分析其优势,并着重讨论如何利用最新的后端技术栈实现一个高效的微服务系统。我们将涵盖设计模式、服务划分、数据一致性、服务发现与注册、API网关以及容器化等关键技术点,为后端开发者提供一份实操指南。 【4月更文挑战第30天】 在数字化时代的浪潮中,移动应用和操作系统的紧密交织已成为日常生活和商业活动的基石。本文将深入探讨移动应用开发的关键技术、跨平台开发工具的选择以及移动操作系统的架构和性能优化策略。通过分析当前移动应用开发的挑战与机遇,我们将
|
15天前
|
安全 Java 数据安全/隐私保护
Spring Boot优雅实现多租户架构:概念与实战
【4月更文挑战第29天】在多租户系统中,一个应用实例服务于多个租户,每个租户享有独立的数据视图,而应用的基础设施被共享。这样的架构不仅优化了资源使用,还能降低维护和运营成本。本文将详细介绍如何在Spring Boot中实现多租户架构,并提供具体的实战案例。
40 2
|
17天前
|
消息中间件 监控 中间件
探索微服务架构下的系统弹性设计
【4月更文挑战第26天】 在当今快速迭代和持续部署的软件发展环境中,系统的弹性设计成为维护高可用性和稳定性的关键因素。本文将深入探讨在微服务架构下如何实现系统弹性,包括识别潜在的故障点、设计容错机制、以及通过实践案例分析提升系统整体的韧性。我们将讨论一系列策略,如服务降级、超时管理、重试策略、断路器模式等,旨在为开发者提供一套实用的系统弹性设计方案。
|
22天前
|
缓存 监控 算法
Python性能优化面试:代码级、架构级与系统级优化
【4月更文挑战第19天】本文探讨了Python性能优化面试的重点,包括代码级、架构级和系统级优化。代码级优化涉及时间复杂度、空间复杂度分析,使用内置数据结构和性能分析工具。易错点包括过度优化和滥用全局变量。架构级优化关注异步编程、缓存策略和分布式系统,强调合理利用异步和缓存。系统级优化则涵盖操作系统原理、Python虚拟机优化和服务器调优,需注意监控系统资源和使用编译器加速。面试者应全面理解这些层面,以提高程序性能和面试竞争力。
18 1
Python性能优化面试:代码级、架构级与系统级优化
|
22天前
|
运维 安全 定位技术
云HIS系统采用B/S架构云端SaaS服务的方式提供,使用用户通过浏览器即能访问
云HIS系统采用B/S架构云端SaaS服务的方式提供,使用用户通过浏览器即能访问
26 2
|
25天前
|
机器学习/深度学习 算法 atlas
RAG 2.0架构详解:构建端到端检索增强生成系统
RAG(检索增强生成)旨在通过提供额外上下文帮助大型语言模型(LLM)生成更精准的回答。现有的RAG系统由独立组件构成,效率不高。RAG 2.0提出了一种预训练、微调和对齐所有组件的集成方法,通过双重反向传播最大化性能。文章探讨了不同的检索策略,如TF-IDF、BM25和密集检索,并介绍了如SPLADE、DRAGON等先进算法。目前的挑战包括创建可训练的检索器和优化检索-生成流程。研究表明,端到端训练的RAG可能提供最佳性能,但资源需求高。未来研究需关注检索器的上下文化和与LLM的协同优化。
78 1
|
27天前
|
消息中间件 运维 Java
B/S架构,采用JAVA编程的医院云HIS系统源码,公立二甲医院应用案例
SaaS模式Java版云HIS系统,在公立二甲医院应用多年,经过多年持续优化系统运行稳定、功能齐全,界面布局合理、操作简便。融合B/S版电子病历系统,支持电子病历四级,HIS与电子病历系统均拥有自主知识产权。 云HIS系统采用云端SaaS服务的方式提供,使用用户通过浏览器即能访问,无需关注系统的部署、维护、升级等问题,系统充分考虑了模板化、配置化、智能化、扩展化等设计方法,覆盖了基层医疗机构的主要工作流程,能够与监管系统有序对接,并能满足未来系统扩展的需要。
B/S架构,采用JAVA编程的医院云HIS系统源码,公立二甲医院应用案例