限界上下文间聚合ID引用的策略:优先解耦而非严格DRY

admin 百科 12

限界上下文间聚合ID引用的策略:优先解耦而非严格DRY

在领域驱动设计中,当一个限界上下文需要引用另一个限界上下文的聚合id时,直接导入id定义会引入不必要的耦合。本文探讨了这种场景下的最佳实践,推荐通过在引用方限界上下文内重新定义id结构来保持各上下文的独立性,即使这违反了dry原则,因为id变更的低频性和高协调成本使得重复的弊端远小于紧耦合的风险。

引言:跨限界上下文ID引用的挑战

在复杂的企业应用中,系统通常被划分为多个限界上下文(Bounded Context),每个上下文负责其特定的领域和业务逻辑。然而,在实践中,一个限界上下文中的实体可能需要引用另一个限界上下文中的聚合根(Aggregate Root)的标识符(ID)。例如,一个“订单”上下文中的订单项可能需要引用“产品”上下文中的产品ID。此时,核心问题在于:我们应该直接导入并重用产品上下文定义的 ProductId 类型,还是在订单上下文内部重新定义一个本地的 ProductId 类型?这涉及到领域驱动设计中“单一职责原则(DRY)”与“限界上下文独立性”之间的权衡。

理解限界上下文与耦合

限界上下文是领域驱动设计中的核心概念,它定义了一个特定领域模型的边界。在这个边界内,领域术语和概念具有明确、统一的含义(即无处不在的语言)。每个限界上下文都应尽可能地保持独立和自治,以降低系统复杂性,促进团队协作,并允许独立演进。

直接从一个限界上下文(如 Context1)导入另一个限界上下文(如 Context2)的类型定义(如 ExampleEntity1ID)会引入显式的代码耦合。这意味着 Context2 现在直接依赖于 Context1 的内部实现细节。这种耦合带来的问题包括:

  1. 脆弱性:如果 Context1 中的 ExampleEntity1ID 定义发生改变,Context2 可能会受到影响,甚至需要修改代码,即使这种改变对 Context2 的业务逻辑没有直接意义。
  2. 部署与测试复杂性:两个上下文的部署和测试不再完全独立,需要协调。
  3. 概念泄漏:Context2 可能会无意中采纳 Context1 的内部概念和语言,模糊了上下文边界,损害了各自无处不在的语言的清晰性。

推荐策略:重新定义ID结构

面对跨限界上下文的ID引用,推荐的策略是在引用方限界上下文内部重新定义(或使用一个简单的原始类型如字符串或UUID)该ID的结构,而不是直接导入。这意味着我们在此场景下选择“打破DRY原则”,以维护限界上下文的独立性。

为什么打破DRY是更优解?

  1. ID的特性:聚合ID通常是简单的值对象,例如一个字符串、UUID或整数。它们的内部结构和行为通常非常稳定,且不包含复杂的领域逻辑。
  2. 变更频率与成本:聚合ID的表示形式(例如,从UUID变为自定义字符串格式)变更的频率极低。当这种变更确实发生时,它通常是一个重大且影响深远的事件,需要跨多个团队和系统的高度协调。在这种情况下,即使存在多处重复的ID定义,变更时也极不可能“遗漏”某个副本,因为整个变更过程会非常谨慎和严格。DRY原则旨在避免因代码重复而导致的维护遗漏,但在ID变更这种特殊情况下,其优势被削弱。
  3. 优先级:在领域驱动设计中,限界上下文的独立性、解耦以及清晰的边界通常比严格遵循DRY原则具有更高的优先级,尤其是在处理简单的、稳定的值对象时。牺牲一点代码重复性,换取更松散的耦合和更清晰的领域边界,是值得的。

代码示例与对比

为了更好地说明这两种方法的差异,我们以Python为例:

假设 Context1 定义了 ExampleEntity1 及其ID:

# domain/context_1/models.py

class ExampleEntity1ID:
    """ExampleEntity1在Context1中的唯一标识符"""
    def __init__(self, value: str):
        if not value:
            raise ValueError("ID value cannot be empty")
        self.value = value

    def __eq__(self, other):
        if not isinstance(other, ExampleEntity1ID):
            return NotImplemented
        return self.value == other.value

    def __hash__(self):
        return hash(self.value)

    def __str__(self):
        return self.value

class ExampleEntity1:
    """Context1中的聚合根"""
    def __init__(self, id: ExampleEntity1ID, some_field_1: str):
        self.id = id
        self.some_field_1 = some_field_1

登录后复制

限界上下文间聚合ID引用的策略:优先解耦而非严格DRY-第2张图片-佛山资讯网

现在,Context2 中的 ExampleEntity2 需要引用 ExampleEntity1ID。

不推荐的做法:直接导入

这种方法直接从 Context1 导入 ExampleEntity1ID。

标签: python ai 为什么 red gate

发布评论 0条评论)

还木有评论哦,快来抢沙发吧~