DDD领域驱动设计基本理解

news2025/1/10 1:47:49

DDD是一种软件设计思想和方法论,以领域为核心构建软件设计体系,将业务模型抽象成领域模型进行拆解和封装。本文简要介绍DDD的基本概念和常用的分层设计架构,并结合业务场景进行领域驱动设计的实战分析,以加深理解。


1、DDD领域驱动设计基本概念理解

1.1 什么是DDD领域驱动设计?

领域驱动设计DDD(Domain Driven Design)是一种从系统分析到软件建模的设计思想和方法论,最早在2004年由Eric Evans在著作《Domain-Driven Design –Tackling Complexity in the Heart of Software》中提出相应的概念。其核心思想是以领域为核心驱动力构建软件设计体系,并围绕业务概念抽象出领域模型,通过领域和边界划分将复杂的业务模型抽象化、简单化,最终实现复杂软件应用系统的拆解和封装。DDD不仅可以用于微服务设计,还可以很好地应用于企业中台的设计,也适用于传统的单体应用。

在DDD中分为战略设计和战术设计:

  • 战略设计:主要从业务视角出发,建立业务领域模型、划分领域边界、建立通用语言的限界上下文。在战略设计中,需要对领域进行全面的了解和分析,探究业务的规则和本质,以及领域未来的趋势发展和可能变化。
  • 战术设计:从技术视角出发,侧重于领域模型的技术实现完成软件开发和落地,包括:实体、值对象、聚合、工厂、仓库、领域服务、领域事件等代码逻辑的设计和实现。战术设计关注的是领域中的具体情境和场景,需要针对具体的问题进行具体的分析和设计,以满足业务需求。
1.2 DDD领域驱动设计和微服务关系

这些年随着软硬件技术的发展,软件设计架构也从最早的单机架构(BS/CS)、到集中式架构,再到如今微服务(MicroServices)分布式架构。

  • 单机架构:采用面向过程的设计方法,系统包括应用客户端层和数据库层,整个系统围绕数据库驱动设计和开发。
  • 集中式架构:采用面向对象的设计方法,经典的三层架构包括业务接入层、业务逻辑层和数据访问层。这种集中式架构容易在某一层变得臃肿,另外受限于集中式的限制,可扩展性差,架构上不灵活。
  • 分布式微服务架构:微服务架构将应用程序拆分为一系列相互协作的微服务,每个微服务负责一个特定的业务领域或业务功能,从而实现应用之间的解耦,同时解决单体应用扩展性和灵活性问题。在微服务架构中,每个微服务都是独立的,可以使用不同的编程语言、数据存储和框架,这使得每个微服务可以独立地演进和部署。

在这里插入图片描述

DDD的概念最早在2004年就提出来,之后一直不温不火直到微服务的概念出现,也可以说DDD的流行和微服务的发展有一定的关系。微服务的兴起使得越来越多的开发人员开始关注如何将应用程序划分成一系列相互协作的微服务,并且每个微服务需要实现自己的业务逻辑。在这个过程中,开发人员发现他们需要一种更好的方式来划分业务领域边界,建立正确的业务模型和通用语言,以便更好地实现微服务的独立性和可维护性。

DDD正是一种能够帮助开发人员更好地设计和实现业务领域的模式。通过将业务领域作为核心,DDD可以帮助开发人员更好地理解业务需求,建立正确的业务模型和通用语言,从而更好地划分微服务边界,从而更好地实现微服务的独立性和可维护性。

1.3 DDD领域驱动设计中基本概念

领域驱动设计DDD中的核心概念如下图所示,包括:

  • 战略设计:包括领域/子域、统一语言、限界上下文等概念
  • 战术设计:使用这些模式来捕获和传递领域中的概念、关系、规则
    • 代表领域的概念:如实体、值对象、领域服务、领域事件等
    • 用于管理对象的生命周期:如聚合、工厂、资源库等
    • 用于集成或跟踪:领域事件、事件溯源等

在这里插入图片描述

1.3.1 统一语言

DDD中的统一语言(Ubiquitous Language)是一种用于描述业务领域中的概念、规则和流程的语言。它是一种由领域专家、开发人员和用户共同理解的通用语言,而不是一种特定的编程语言或框架。

统一语言贯穿于整个开发过程,从需求分析到设计再到编码。它应该用于描述业务领域中的实体、值对象、聚合、领域服务等概念,同时也要用于描述业务规则、流程和交互。
在统一语言中,应该使用业务领域的术语和词汇,而不是技术术语和词汇。例如,应该使用“客户”而不是“用户”来描述应用程序的客户端,因为“客户”是业务领域中的术语,而“用户”则是技术领域的术语。

统一语言的重要性在于,它能够帮助开发人员更好地理解业务需求和领域模型,同时也能帮助业务专家更好地理解技术和系统实现。它还能够提高代码的可读性和可维护性,因为代码中的术语和词汇能够被业务专家和开发人员共同理解。

1.3.2 领域和子域

在研究和解决业务问题时,DDD会按照一定的业务规则将业务领域进行细分,当领域细分到一定的程度后,DDD会将问题范围限定在特定的边界内,在这个边界内建立领域模型。简言之,领域是业务领域的范围和边界。

领域可以进一步划分为不同的子领域称为子域,每个子域都包含了一些特定的业务规则和功能,它们共同构成了整个业务领域。领域和子域是基于业务需求和功能进行划分,在划分领域和子域时,需要考虑到业务需求的变化和扩展,使得系统能够更好地适应业务需求的变化。

领域和子域的划分也是一种设计上的分层,它能够帮助开发人员更好地理解和组织系统中的代码和模型。在实现时,每个子域都可以被划分成一个或多个限界上下文(Boundary Context),每个限界上下文都包含了一些特定的业务规则和功能,它们共同实现了该子域的功能。

1.3.3 核心域、通用域和支撑域

领域在不断划分的过程中,细分为不同的子域,根据重要性的不同,子域又划分为核心域、通用域和支撑域。

  1. 核心域是指业务领域中的核心业务逻辑和模型,它们是领域驱动设计的核心部分。核心域包含领域模型、领域服务、业务规则等,它们是业务领域中最重要、最稳定的部分。
  2. 通用域是指业务领域中通用的、可复用的业务逻辑和模型,它们不是核心业务逻辑,但可能在多个核心域中被使用。通用域包含一些通用的业务逻辑、数据校验、安全控制等。
  3. 支撑域是指用于支持业务领域中的核心域和通用域的基础设施和工具,包括数据存储、消息传递、分布式系统、自动化测试等。支撑域中不包含业务逻辑和模型,而是聚焦于技术实现和系统监控、管理等方面。
1.3.4 界限上下文

限界上下文是指一个业务领域的边界,它包含了多个子域(Subdomain),每个子域都包含了一些特定的业务规则和功能。限界上下文用来封装通用语言和领域对象,提供上下文环境,保证在领域之内的一些术语、业务相关对象等(通用语言)有一个确切的含义,没有二义性。

限界上下文是一个显式的边界,领域模型便存在于这个边界之内;在边界内,通用语言中的所有术语和词组都有特定的含义,而模型需要准确地反映通用语言。通过限界上下文,开发人员可以更好的组织和管理领域中的不同部分,更好的理解业务需求和领域模型。

领域、子域和限界上下文的关系如下图所示:

在这里插入图片描述

1.3.5 实体Entity

实体是拥有唯一标识和状态,且具有生命周期的业务对象。在实体的生命周期内,无论其如何变化,其仍旧是同一个实体。实体的标识和状态可以在业务过程中发生变化,而实体之间的关系也可以随着业务过程的变化而发生变化。

  • 实体的业务形态:在领域模型中实体是多个属性、操作或行为的载体。实体的属性描述实体的状态比如客户的姓名、年龄等;实体的行为指实体可以执行的操作,比如创建、更新等。实体和值对象是组成领域模型的基础单元。
  • 实体的代码形态:在代码模型中,实体表现为实体类,这个类包含了实体的属性和方法,通过这些方法实现实体自身的业务逻辑。
  • 实体的运行形态:实体是以DO(领域对象)的形式存在,每个实体对象都有一个唯一的ID,这个ID在实体的整个生命周期内无论实体对象怎么变化都具有唯一性。
  • 实体的数据库形态:DDD先构建领域模型,再针对业务场景构建实体对象和行为,最后将实体对象映射到数据持久化对象。在领域模型映射到数据模型时,一个实体可能对应0个、1个或者多个数据库持久化对象。
1.3.6 值对象

《实现领域驱动设计》一书中对值对象的定义如下:

通过对象属性值来识别的对象,它将多个相关属性组合为一个概念整体。在DDD中用来描述领域的特定方面,并且是一个没有标识符的对象,叫作值对象。值对象没有唯一标识,没有生命周期,不可修改,当值对象发生改变时只能替换。

值对象将不同的相关属性组成一个整体概念,是一堆不可变属性的集合。

  • 值对象的业务形态:值对象是若干不可修改的属性集合,只有数据初始化操作和有限不涉及数据修改的行为,基本不包含业务逻辑。值对象在逻辑上依然是实体属性的一部分
  • 值对象的代码形态:如果值对象是单一属性例如字符串、整型、枚举等,则直接定义为实体类的属性;如果值对象是属性集合,则把它设计为Class类,包含多个属性。
  • 值对象的运行形态:值对象嵌入到实体中,有两种不同的数据格式分别是属性嵌入的方式和序列化大对象的方式,比如JSON字段
1.3.7 聚合和聚合根

1)聚合和聚合根

DDD中实体和值对象是很基础的领域对象,实体一般对应业务对象、值对象一般是属性集合,但是实体和值对象都是个体化的表现。而聚合是把一些关联性极强、生命周期一致的实体、值对象放到一个聚合里。聚合根(Aggregate Root)是指聚合中的实体,它负责聚合的内聚性和一致性,同时也是聚合的入口点。

聚合有一个聚合根和上下文边界,这个边界根据业务单一职责和高内聚原则,定义了聚合内部应该包含哪些实体和值对象,而聚合之间的边界是松耦合的。按照这种方式设计出来的微服务很自然就是“高内聚、低耦合”的。

在这里插入图片描述

2)聚合设计原则

在领域驱动设计中,聚合设计是非常重要的一部分。以下是聚合设计的一些原则

  • 保持完整性:聚合中包含一组相互关联的实体和值对象,它们被视为一个整体。聚合的边界应该明确,以保持其完整性。
  • 保持一致性:聚合内数据强一致性,而聚合之间数据最终一致性。聚合中的实体和值对象在业务规则和约束下保持一致。如果一个聚合违反了业务规则或约束,那么它就不是一个一致的聚合。
  • 保持小型聚合:聚合尽可能小,只包含必要的实体和值对象。小型聚合可以降低由于业务过大导致聚合重构,有助于提高系统的可维护性和可扩展性。
  • 避免聚合根以外的实体:聚合根以外的实体应该尽可能避免,因为它们增加了系统的复杂性和耦合度。如果必须使用这些实体,应该确保它们与聚合根之间的关系是一致的。
  • 避免跨聚合引用:如果一个实体或值对象需要引用另一个聚合中的实体或值对象,通过关联外部聚合根ID的方式引用,而不是直接对象引用的方式。这样可以保持聚合的独立性和可维护性。
1.3.8 工厂

如果实体或值对象的创建过程非常复杂,可以将其委托给工厂。DDD中工厂是指用于创建领域模型对象的工厂方法,它用于将业务逻辑和实现技术解耦。它是一种设计模式,允许开发人员通过调用工厂方法来创建对象,而不需要直接在代码中实例化对象。

将创建复杂对象和聚合的职责分配给一个单独的对象,该对象本身并不承担领域模型中的职责,但是依然是领域设计的一部分。工厂应该提供一个创建对象的接口,该接口封装了所有创建对象的复杂操作过程,同时,它并不需要客户去引用那个实际被创建的对象。对于聚合来说,我们应该一次性地创建整个聚合,并且确保它的不变条件得到满足。

1.3.9 仓库

仓库(Repository)是一种模式,用于封装数据访问逻辑,提供对数据的持久化和查询。仓库操作的最小单元就是聚合,每个聚合会对应一个仓库。它旨在将数据访问细节与领域模型分离,使领域模型更加独立和可测试。

在这里插入图片描述

仓库提供了一种统一的接口,使得领域模型可以与不同的数据存储方式进行交互,同时也提供了一些查询操作。例如,一个客户仓库可以具有存储客户、检索客户等方法。通过调用仓库方法,开发人员可以存储和检索客户对象,同时也可以在存储和检索对象时执行其他的业务逻辑和验证。

1.3.10 领域服务

领域服务(Domain Service)是指跨越多个聚合或子域的逻辑服务,用于处理业务需求和行为。领域服务通常用于执行跨聚合的操作,例如处理业务规则、集成外部系统、执行复杂业务流程等。

需要注意的是,实体和领域服务在实现业务逻辑上不是同级的,当领域中的某些功能,单一实体或者值对象不能实现时,领域服务就会出马,它可以组合聚合内的多个实体或者值对象,实现复杂的业务逻辑。

1.3.11 领域事件

领域事件(Domain Event)是指业务领域中发生的一些事件,通常意味着领域对象状态的改变。领域事件在系统中起到了传递消息、触发其他动作的作用,是解耦领域模型的重要手段之一。在实现时,领域事件可以由领域模型中的实体或服务触发,例如当用户注册成功时,可以触发一个“用户注册成功”的领域事件。领域事件也可以由外部系统或服务触发,例如当第三方支付系统支付成功时,可以触发一个“订单支付成功”的领域事件。

领域事件处理包括:事件构建和发布、事件数据持久化、事件总线、消息中间件、事件接收和处理等。

在这里插入图片描述

  • 事件构建和发布:
    • 事件基本属性通常包含事件的标识符、时间戳和事件数据等信息;业务属性用于记录事件发生时刻的业务数据。事件的基本属性和业务属性一起构成事件实体。
    • 事件发布方式很多,可以通过应用服务或领域服务发布到事件总线或消息中间件;也可以从事件列表中定时获取增量事件数据发布到消息中间件。
  • 事件数据持久化:用于系统之间的数据对账,或者实现发布方和订阅方数据的审计
    • 持久化到本地业务数据库的事件表中,利用本地事务保证业务和事件数据的一致性
    • 持久化到共享的事件数据库中,事件数据库和业务库是独立的。
  • 事件总线:实现微服务内聚合之间领域事务,提供事务分发和接收等服务
  • 消息中间件:跨微服务的领域事件大多会用到消息中间件,实现跨微服务的事件发布和订阅
  • 事件接收和处理:订阅方采用监听机制接收消息队列中的事件数据,当完成事件数据的持久化处理后,就可以进行下一步的业务处理
1.3.12 贫血模型和充血模型

1)贫血模型(Anemic Domain Model)

所谓的贫血模型指的是定义领域对象的时候,只有对象的属性信息,没有对象的行为信息。贫血模型强调了业务逻辑和数据之间的分离,将业务逻辑和数据分别放在不同的对象中,导致领域模型变得贫血、不灵活和难以维护。

2)充血模型(Rich Domain Model)

充血模型指的是领域对象既包括属性信息,又包含行为信息。在充血模型中,领域对象包含了业务逻辑、数据和规则,它们被组织在一个聚合根中。
相对而言贫血模型相对简单,模型上只有数据没有行为,针对简单的模型可以快速的完成交付,但是后期的成本较高,并且难以保证事务的一致性和完整性。充血模型则是领域模型模式,实现逻辑上由各自的对象负责,具备事务的一致性和完整性,事务边界也被包含在聚合根中。

1.4 DDD领域驱动设计中架构模型

在DDD领域中,常见的领域驱动设计架构有以下几种:

  1. 分层架构:将领域模型和业务逻辑分离出来,并减少对基础设施和应用层的依赖,从最早三层架构,演化到四层架构、五层架构,包括用户接口层、应用层、领域层和基础设施层。分层架构中,每层只能与位于其下方的层耦合。
  2. 六边形架构:也称为端口与适配器。对于每种外界类型,都有一个适配器与之相对应。该架构的定义涉及六个部分,分别是应用程序、域、基础设施、适配器、API和外部系统。六边形架构的核心理念是应用通过端口与外部进行交互。
  3. CQRS架构:CQRS架构是一种读写分离的架构设计,系统被分为两个部分:命令端(Command)和查询端(Query)。命令端负责处理写操作(例如修改数据),而查询端则负责处理读操作(例如查询数据)。
  4. 事件驱动架构:用于处理事件的生成、发现和处理等任务的软件架构。在这种架构中,领域对象之间通过发布和订阅事件来通信,一个事件通常会引发多个服务的同步,从而引起事件的扩散。
1.5 DDD领域驱动设计的优缺点

实现DDD的最大好处是第一时间将业务需求构建成领域模型,而不是将其切割为数据和行为,然后再用数据库去实现。DDD具有以下优点:

  • 统一业务语言:通过使用统一的领域语言,准确传达业务规则,提升团队间的沟通效率。
  • 清晰业务边界:统一各个子域的认识,通过领域模型界定需求实现边界。
  • 提升变化应对:通过领域模型与数据模型的分离,将核心业务的不变与需求的变进行有效隔离,提升架构应变化的能力。

当然使用DDD需要一定的学习和使用成本,开发人员需要掌握一定的领域建模知识和技能。同时在高并发场景下,由于DDD使用的模型可能会增加一些额外的开销。

2、DDD领域驱动设计主要架构模型介绍

2.1 分层架构

在这里插入图片描述

DDD的分层架构在不断的优化,在最早的四层架构中(如左上图所示),基础设施层被其它层依赖,处于最核心的位置,但实际上领域层才是依赖的核心。后来采用依赖倒置的设计,优化了传统的四层架构,实现了各层对基础设施层的解耦。四层架构包括:用户接口层、应用层、领域层和基础设施层。

  1. 用户接口层:负责展示用户界面和处理用户输入,它通常包含一些前端框架和UI组件
  2. 应用层:负责处理业务逻辑和应用程序的流程,它通常包含一些应用程序服务,例如事务处理、规则验证和流程管理等
  3. 领域层:领域层是领域模型的核心,它负责表示业务领域中的实体、值对象、聚合、领域服务等。领域层通常包含一些领域模型类、领域服务类和领域事件等。
  4. 基础设施层:负责提供一些通用服务和功能,例如数据存储、消息传递、日志记录等。基础设施层通常包含一些工具类、库和框架等。

在传统架构中,由于上层应用对数据库的强耦合,很多业务在架构演进中担心更换数据库对应用的影响,因为一旦更换数据库,就可能需要重写大部分代码,这对开发来说是致命的。采用依赖倒置的设计后,应用层通过解耦保持独立的核心业务逻辑,当数据库变更后,只需要更换数据库基础服务即可,对应用的影响降到最低。

2.2 六边形架构

六边形架构又称为“端口-适配器”架构,对于每种外界类型,都有一个适配器与之相对应,其核心设计思想是外界通过应用API与内部进行交互。

在这里插入图片描述

如图所示的六边形架构中,核心业务逻辑(应用程序和领域模型)与外部资源(包括Web应用、基础设施等)完全隔离,仅通过适配器进行交互。除了核心的领域模型,还包括如下三个层次:

  • 应用程序层(Application Layer):负责处理业务逻辑和应用程序的流程,它通常包含一些应用程序服务,例如事务处理、规则验证和流程管理等。
  • 适配器层(Adapter Layer):负责与外部系统进行交互,例如与数据库、消息队列、Web 服务等进行通信和交互。它通常包含一些适配器类和工具类等。
  • 基础设施层(Infrastructure Layer):负责提供一些通用服务和功能,例如数据存储、消息传递、日志记录等。基础设施层通常包含一些工具类、库和框架等。

在六边形架构中,一个端口可能对应多个外部系统,不同的外部系统也可能会使用不同的适配器,由适配器负责协议转换。因此在使用时,根据用例来设计应用程序,所有的适配器使用相同的API满足不同的端口访问需求。

2.3 CQRS架构

CQRS是一种读写分离的架构设计,系统分为两个部分:命令端和查询端,命令端负责处理写操作,而查询端负责处理读操作。在CQRS架构中,写操作和读操作被分离并分布在两个独立的系统中。命令端和查询端之间的数据一致性采用最终一致性的方式处理。这意味着查询端可能会返回不是最新的数据,但系统会确保在一定时间内数据达到一致状态。

在这里插入图片描述

除了分离写操作和读操作,CQRS架构还涉及到事件的概念。事件表示命令操作领域中的聚合根状态发生变化后产生的事件。这些事件可以被其他服务订阅,以实现系统的联动。

3、DDD领域驱动设计落地实践

前面介绍了DDD的基本概念以及主要架构,这些概念很抽象,读起来是一头雾水。下面以“在线书店”为例,探讨下DDD具体落地实践。

3.1 战略设计

在战略设计阶段,主要过程分为业务场景分析、领域建模、划分边界上下文,战略设计是DDD中的核心步骤。

3.1.1 业务场景分析

业务场景分析阶段是对业务的全面梳理,将业务中涉及到的细节展现出来,为后续的领域建模提供足够的业务输入。常见的业务场景分析方法有事件风暴法、四色建模法、用例分析法等,四色建模法相对比较复杂,本文以事件风暴法对“在线书店”业务场景进行分析梳理。

1)场景分析参与方

事件风暴类似于头脑风暴的方法,业务场景的分析讨论参与角色包括领域专家、产品经理、需求分析人员、架构师、项目经理、开发经理和测试经理。参与者要尽可能遍历所有业务细节,列出具体的业务场景、业务规则、业务之间的关联以及热点问题等。

2)业务讨论与梳理

经过事件风暴讨论后,总结出“在线书店”下的典型的业务场景,包括:

  • 书籍管理:核心业务场景是书籍销售,书籍管理包括书籍信息的录入、编辑、查询、分类、推荐等。
  • 用户管理:为用户提供注册、登录、个人信息管理、购物车管理等功能。
  • 订单管理:为用户提供订单的创建、支付、发货、跟踪等功能。
  • 支付管理:为用户提供安全的支付方式

分析业务中的事件,寻找业务逻辑和业务规则,比如用户购书流程,从用户注册登录、书籍查询推荐、订单创建和支付等流程进行梳理。下面是按照业务操作对象、事件和命令(动作)形成事件风暴小黑板对整个业务流程进行梳理。

在这里插入图片描述

3.1.2 领域建模

在领域建模时,根据场景分析过程中产生的领域对象,找出产生命令的实体,分析实体之间的依赖关系产生聚合,为聚合划定界限上下文,建立领域模型以及模型之间的依赖。根据上图业务场景分析,提取产生这些行为的实体,包括用户、订单、书籍、购物车和系统管理。

在这里插入图片描述

完成领域对象分析后,需要构建业务聚合,首先在实体中找到聚合根,根据聚合根的特点,聚合根是一个实体具有全局唯一标识,并且具备生命周期的同时需要专门的模块进行管理。根据这些特征就有了书籍、用户、订单和系统管理这些聚合。

3.1.3 划分边界上下文

获得了整个业务流程中的所有聚合后,需要更具业务语义上下文将具体的聚合划分到对应的上下文中。根据订单和购物车的业务操作行为,订单和购物车共同构成订单域分别管理订单和购物车,用户管理和系统管理共同构成系统和用户管理,因此分为订单域、商品域及系统和用户管理域三个限界上下文。

在这里插入图片描述

3.2 战术设计
3.2.1 微服务拆分

通常一个限界上下文拆分为一个微服务,微服务的设计还需要考虑服务的粒度、分层、边界划分、依赖关系和集成关系。因此,在“在线书店”业务场景中,拆分为订单服务、商品服务和系统用户管理服务三个微服务。

3.2.2 领域分层

根据前面介绍的四层分层架构,对领域进行分层设计,包括:用户接口层、应用层、领域层和基础设施层。

//用户接口层
+-- UI接口

// 领域层  
+-- Entity (实体)  
+-- Value Object (值对象)  
+-- Domain Service (领域服务)  
+-- Domain Rule (领域规则)  
+-- Domain Event (领域事件)  
+-- Resource Library (资源库)  
  
// 应用层  
+-- Application Service (应用服务)  
+-- Application Event (应用事件)  
+-- Application Command (应用命令)  
+-- Application Query (应用查询)  
  
// 基础层  
+-- Data Access Object (数据访问对象)  
+-- Repository (仓储)  
+-- Service Location (服务定位)  
+-- Logging (日志记录)  
+-- Data Store (数据存储)
3.3 代码实现

1)定义书籍实体,它具有书名、作者、出版日期和价格等属性

class Book:  
    def __init__(self, title, author, publication_date, price):  
        self.title = title  
        self.author = author  
        self.publication_date = publication_date  
        self.price = price

2)定义一个值对象类ISBN,它代表书的唯一标识符

class ISBN:  
    def __init__(self, value):  
        self.value = value

3)定义一个领域服务类BookstoreService,它包含一些业务逻辑,例如添加书籍、查找书籍和下架书籍等

class BookstoreService:  
    def add_book(self, book):  
        # 在这里添加书籍到数据库或者其他数据存储中  
        pass  
  
    def find_book_by_title(self, title):  
        # 在这里根据书名查找书籍  
        pass  
  
    def remove_book(self, isbn):  
        # 在这里根据ISBN删除书籍  
        pass

4)定义一个领域事件类BookEvent,它代表书籍状态的变化,例如书籍上架、书籍下架等:

class BookEvent:  
    def __init__(self, book, event_type):  
        self.book = book  
        self.event_type = event_type

5)定义一个应用服务类BookstoreApplicationService,它与用户交互,并调用领域服务类中的方法来实现业务逻辑

class BookstoreApplicationService:  
    def __init__(self, bookstore_service):  
        self.bookstore_service = bookstore_service  
  
    def add_book(self, title, author, publication_date, price):  
        book = Book(title, author, publication_date, price)  
        self.bookstore_service.add_book(book)  
        book_event = BookEvent(book, "added")  
        # 在这里发布书籍事件  
        pass  
  
    def find_book_by_title(self, title):  
        book = self.bookstore_service.find_book_by_title(title)  
        if book:  
            return book  
        else:  
            raise ValueError(f"Book with title '{title}' not found")  
  
    def remove_book(self, isbn):  
        book = self.bookstore_service.remove_book(isbn)  
        book_event = BookEvent(book, "removed")  
        # 在这里发布书籍事件  
        pass

以上是Python实现一个简单的DDD项目示例,它包含了实体、值对象、领域服务、领域事件和应用服务等多个组件。

4、总结

本文简要介绍了DDD领域驱动设计的基本概念、常用的分层架构,并结合实际业务场景落地DDD设计实现。总而言之,DDD是一种软件设计的思想和方法论,其核心是将业务抽象成领域模型,并以领域为核心驱动力构建软件设计体系,将复杂的业务模型抽象化,非常适合微服务架构下的开发设计。


参考资料:

  1. 《实现领域驱动设计》,Vaughn Vernon著
  2. 《解构领域驱动设计》,张逸著
  3. 领域驱动设计DDD|从入门到代码实践,阿里云开发者
  4. https://blog.csdn.net/m0_73311735/article/details/127037315
  5. https://blog.csdn.net/m0_37583655/article/details/117565641
  6. DDD兴起的原因以及与微服务的关系
  7. 2万字带你入门DDD,芋道源码
  8. DDD 领域驱动设计落地实践:六步拆解DDD,慕枫技术笔记

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

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

相关文章

opencv通过轮廓去除虚线

思路: 将虚线膨胀为实线,通过高度和宽度找到轮廓,再将轮廓内的面积涂白色 img cv2.imread(imagePath) gray cv2.cvtColor(img, cv2.COLOR_BGR2GRAY) gray_test gray.copy() binary_test cv2.adaptiveThreshold(clean_gray(gray_test),25…

做项目,最难搞定的不是甲方爸爸...

早上好,我是老原。 前几天和一个老朋友吃饭的时候,他和我吐槽他上个月做的一个项目,实在太累了,几乎是没日没夜地赶进度,身体都快垮了。 我问他既然时间来不及,为什么不前期就和客户沟通好。 他说其实客…

uni-app 的使用体验总结

框架简介 uni-app 是一个使用 Vue.js (opens new window)开发所有前端应用的框架,开发者编写一套代码,可发布到iOS、Android、Web(响应式)、以及各种小程序(微信/支付宝/百度/头条/飞书/QQ/快手/钉钉/淘宝)…

【自然语言处理】COLD:中文攻击性言论检测数据集

COLD:A Benchmark for Chinese Offensive Language Detection 文章目录 COLD:A Benchmark for Chinese Offensive Language Detection1 论文出处2 背景2.1 背景介绍2.2 针对问题2.3 创新点 3 数据集构建3.1 数据源3.2 效率改进3.3 数据集分析 4 实验设计…

驱动开发:内核物理内存寻址读写

在某些时候我们需要读写的进程可能存在虚拟内存保护机制,在该机制下用户的CR3以及MDL读写将直接失效,从而导致无法读取到正确的数据,本章我们将继续研究如何实现物理级别的寻址读写。 首先,驱动中的物理页读写是指在驱动中直接读…

LiveGBS流媒体平台GB/T28181功能-海康大华宇视华为NVR等4G摄像头自带物联网卡注册国标平台后看不到设备的时候如何排查及抓包

LiveGBS流媒体平台GB/T28181功能-海康大华宇视华为NVR等4G摄像头自带物联网卡注册国标平台后看不到设备的时候如何排查及抓包 1、设备注册后查看不到1.1、是否是4G|5G摄像头1.2、关闭萤石云1.3、防火墙排查1.4、端口排查1.5、IP地址排查1.6、设备TCP/IP配置排查1.7、设备多网卡…

【Nexus】Maven从Nexus中下载jar包

目录 一、前言二、配置Apache Maven1、在Maven的settings.xml中添加一个镜像配置,并覆盖中央仓库的默认配置 二、创建Maven项目,配置pom文件拉取Nexus中的jar包1、确定配置的Maven的settings.xml是否是上一步修改的settings.xml文件,以及repo…

规划地类、用途分区、空间管制区代码对应表

规划地类、用途分区、空间管制区代码对应表 —the—end—

UE5 与 C++ 入门教程·第一课:角色与 Enhanced Input

本文主要围绕 UE5 新的输入系统,手把手从 0 搭建 Unreal 项目,实现角色的基础移动。 重要提示:众所周知,C 属于编译型语言,因此动态灵活性不足,不过执行效率高,而蓝图简单灵活,却执行…

探索TCC:释放高可用性和弹性事务的潜力

1、TCC简介 分布式事务是指在分布式系统中,多个服务之间需要保证数据的一致性和完整性的场景。传统的单机事务无法满足分布式系统的需求,因此需要引入一种新的事务模型来解决分布式事务问题。 TCC(Try-Confirm-Cancel)是一种基于…

MySQL的分库分表

分必要不要分库分表(通过优化之后还明显影响业务再分,可以通过监控慢查询确定) 分库分表的一般条件:单表数据量超过1000w(阿里应该是说5000w)或者单表数据文件(.ibd)超过20GB,这个很重要,&…

点云配准综述一篇综述《A comprehensive survey on point cloud registration》(翻译)

参照了 2021最新关于点云配准的全面综述 - 知乎,并且加了些自己翻译,全篇的内容可能稍有删减。主要作为个人笔记,阅读了几篇综述,发现这篇是质量较好的,值得花时间细读。 文章分类 文章将配准方法分为了同源配准和不…

JMeter三大重要组件——线程组、取样器、查看结果数(3)

JMeter三大重要组件 一、JMeter三大重要组件——线程组1、作用:JMeter主要通过线程组来运行用户脚本2、在取样器错误后要执行的动作:3、线程属性3、调度器4、setUp线程组和tearDown线程组 二、JMeter三大重要组件——取样器1、基本a、自动重定向和跟随重…

Obsidian多端同步插件LiveSync

网友 Leo 和 Paco反馈,群晖升级到 DSM7.2 ,注册表可以搜索镜像,根据 Leo 贴的 /var/packages/Docker/etc/dockerd.json 的内容,DSM7.2 应该是使用了 https://docker.nju.edu.cn 作为注册表镜像,但老苏测试过下面几种情…

易基因:易基因近期染色质免疫共沉淀测序(ChIP-seq)研究成果|项目集锦

大家好,这里是专注表观组学十余年,领跑多组学科研服务的易基因。 在生物学研究中,DNA与蛋白质之间的互作(DNA-Protein Interactions,DPIs)是至关重要的,参与基因的表达、调控、复制、重组和修复…

m 序列(最长线性反馈移位寄存器序列)详解

本专栏包含信息论与编码的核心知识,按知识点组织,可作为教学或学习的参考。markdown版本已归档至【Github仓库:https://github.com/timerring/information-theory 】或者公众号【AIShareLab】回复 信息论 获取。 文章目录 m 序列 (最长线性反…

Git进阶系列 | 8. 用Reflog恢复丢失的提交

Git是最流行的代码版本控制系统,这一系列文章介绍了一些Git的高阶使用方式,从而帮助我们可以更好的利用Git的能力。本系列一共8篇文章,这是最后一篇。原文:Using the Reflog to Restore Lost Commits[1] “Reflog”是Git不太为人所…

常见的未授权漏洞批量检测工具

常见的未授权漏洞检测 命令行版已放出支持多线程,批量扫描,指定服务扫描,命令行版地址https://github.com/xk11z/unauthorized_com GUI版unauthorizedV2已更新,可批量ip检测导出结果 项目包含 1 、FTP 未授权访问&#xff08…

SadTalker AI模型使用一张图片与一段音频便可以自动生成视频

SadTalker模型是一个使用图片与音频文件自动合成人物说话动画的开源模型,我们自己给模型一张图片以及一段音频文件,模型会根据音频文件把传递的图片进行人脸的相应动作,比如张嘴,眨眼,移动头部等动作。 SadTalker,它从音频中生成 3DMM 的 3D 运动系数(头部姿势、表情),…

webstorm配置vue开发环境

🌳🌳🌳前言:本文章针对于如何用IDE和webstorm运行一个别人的vue项目进行步骤记录。 📙参考:(10条消息) idea配置vue开发环境_idea配置vue运行环境_drinkworld的博客-CSDN博客https://blog.csdn.net/drinkwo…