十年开发老司机,感悟DDD领域驱动设计的战略设计到底是什么?

模型设计,DDD 分两阶段,战略设计和战术设计

战略设计

战略设计的子域、限界上下文和上下文映射图等概念大致分为:


  • 业务划分
    以区分不同业务,即划分识别出来的业务概念
  • 落地成解决方案
    将划分出来的业务落地

业务概念的划分

首先需要明确:

问题是什么

我们要解决的问题就是领域问题,相关概念如,子域、核心域、支撑域、通用域等。其实都是:如何分解问题。

如何解决

领域(Domain)是一号位,对应待解决的问题。解决问题通常思路是分而治之,DDD就把一个大领域分解成若干小领域(子域(Subdomain))。


DDD首先要建立起一套通用语言,拥有一致的业务词汇表,它们对应模型。

接着,要分类词汇,即把它们划分到不同子域。关键就是分离关注点。


比如,做个项目管理软件,就要有用户、项目、团队,不同人还要扮演不同角色。

第一步,至少先分开身份管理、项目管理,因为关注点不同:


  • 身份管理,关注用户身份信息,如用户名、密码
  • 项目管理,关注项目和团队等


这就有了俩子域:身份管理,项目管理。


若直接给结果,你可能会觉得很好理解。但划分出不同子域还是容易出问题的,因为有些概念不易区分。

比如,用户应该怎么划分?放在身份管理是合适的,但项目管理也要用到呀!


根据单一职责原则,它给了我们一个重要思考维度,变化从何而来

不同角色的人关注不同变化,所以,虽然我们用的词都是“用户”,但想表达的含义其实不同,最好分开这些不同的含义,即分开不同角色:

业务概念的落地

DDD问题层面的概念已经阐述完毕。接下来,就是解决方案层面。


切分出的子域,怎样落实到代码?

首先要解决的就是这些子域如何组织?


  • 写一个程序把所有子域都放里面
  • 每个子域单独做个应用
  • 有一些在一起,有一些分开


这就引出DDD的

限界上下文(Bounded Context)

限定了通用语言*使用的边界,一旦出界,含义便无法保证。

比如,同样是“订单”,不加限制,很难区分它在哪种场景。一旦定义了限界上下文,那交易上下文的“订单”和物流上下文的“订单”肯定不同。就是因为订单这个说法,在不同边界内,含义不同。


注意,子域和限界上下文不一定是一一对应的,可能在一个限界上下文中包含了多个子域,也可能在一个子域横跨了多个限界上下文。


限界上下文是在解决方案层面,所以,就可以把限界上下文看作一个独立系统。限界上下文与微服务的理念契合,每个限界上下文都可成为一个独立服务。


限界上下文是完全独立的,不会为了完成一个业务需求要跑到其他服务中去做很多事,这恰是很多微服务出问题的点,比如一个业务功能要调用很多其他系统功能。


有限界上下文,就可以把整个业务分解到不同的限界上下文中,但是,尽管我们拆分了系统,它们终究还是一个系统,免不了交互。

比如:


  • 一个用户下了订单,这是在订单上下文中完成的
  • 用户要去支付,这是在支付上下文中完成的


我们要通过某种途径让订单上下文的一些信息发送到支付上下文。所以,要有一种描述方式,描述不同限界上下文之间交互的方式-上下文映射图(Context Map)。

DDD 给我们提供了一些描述这种交互的方式,比如:

  • 合作关系(Partnership)
  • 共享内核(Shared Kernel)
  • 客户-供应商(Customer-Supplier)
  • 跟随者(Conformist)
  • 防腐层(Anticorruption Layer)
    防腐层是最具防御性的一种关系,就是在外部模型和内部模型之间建立起一个翻译层,将外部模型转化为内部模型。但凡有可能,就要建立防腐层,将外部模型完全隔离开。
  • 开放主机服务(Open Host Service)
  • 发布语言(Published Language)
  • 各行其道(Separate Ways)
  • 大泥球(Big Ball of Mud)
    要规避


这么多交互方式,主要是为让你在头脑中仔细辨认,看看限界上下文之间到底在以怎样的方式交互。


知道不同限界上下文之间交互方式后,不同交互方式就可落地为不同协议。

常用协议如:REST API、RPC 或是 MQ, 按需选型即可。


在我们定义好不同的限界上下文,将它们之间的交互呈现出来之后,就得到了一张上下文映射图。

上下文映射图是可以帮助我们理解系统的各个部分之间,是怎样进行交互的,建立全局性认知。


上一篇:供应链商品域DDD实践


下一篇:关于领域驱动设计(DDD)中聚合设计的一些思考