Should DDD entities compare by reference or by ID?

匿名 (未验证) 提交于 2019-12-03 03:02:02

问题:

When I started using DDD, I created Equals() methods in my entities that compared the ID of the entity. So two entity objects with the same ID would be considered equal.

At some point I thought about that and found that two entities in different states should not be considered equal, even when they describe the same thing (i.e. have the same ID). So now I use reference equality for my entities.

I then stumbled over this answer by Mark Seemann, where he writes

Entities are equal if their IDs equal each other.

Now, of course, I'd like to know which approach is better.

Edit: Note that the question is not whether having two instances of the same entity at the same time is a good idea. I'm aware that in most situations it is probably not.

回答1:

The question is twofold. First, what you really want to know is

How to handle terms that the X language (or Y framework) impose when I code a domain model with it?

C# for example imposes you that any new concept you define inherit a certain set of public methods. Java includes even more methods.

I've never heard a domain expert talking about hash codes or instance equality, but this is one of those situations when the (often misunderstood) quote "don't fight the framework" from Evans apply: just teach developers to not use them when they do not belong to domain's interfaces.

Then, what you want to know is

What is an entity? How it relates to its own identity?

Start with why! You know that entities are terms of the ubiquitous language that are identifiable.

But WHY?

Plain simple: entities describe concepts whose evolution in time is relevant in the context of the problem we are solving!

It is the relevance of the evolution that defines the entity, not the other way around! The identity is just a communication tool to keep track of the evolution, to talk about it.

As an example think about you: you are a person with a name; we use your name to communicate about your interactions with the rest of the world during your life; still, you are not that name.

Ask yourself: why I need to compare domain entities? Is the domain expert talking this way? Or I'm just using a DDD parlance to describe a CRUD application that interact with a relational database?
To me, the need to actually implement Equals(object) or GetHashCode() into an entity looks like a smell of an inadequate infrastructure.



回答2:

Entities shouldn't be compared like that, in the first place. There is no valid use case (outside testing, but then again the assertion library should handle this for you) to see if 2 entities are equal using the object's Equals method.

What makes an Entity unique is its Id. The purpose of the id is to say 'this very entity is different from other entities despite having identical properties/values'.

That being said, in a Domain you might need to compare a concept instance with another instance. The comparison is done according to Bounded Context (or even Aggregate) specific Domain rules. It doesn't matter an entity is involved, it could have been a value object as well.

Basically the 'comparison' should be a Domain use case which will be probably implemented as a service. This has no relation to an object's Equals method, which is a technical aspect.

When doing DDD, don't think like a programmer (i.e technical aspects) think like an architect (high level). Code, programming language etc is just an implementation detail.



回答3:

I think it's bad idea to have the two separate instances of the same entity in different states. I can't think of a scenario where that would be desirable. Maybe there is one? I believe there should only ever be one instance of a given entity with a particular ID.

In general I'd compare their equality using their IDs.

But if you wanted to check if they are the same object reference then you could just use:

        if (Object.ReferenceEquals(entityA, entityB))         {             DoSomething();         } 


标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!