我正在使用CQRS和域事件概念构建具有域模型的应用程序(但没有事件源,只是普通的旧SQL)。 SomethingChanged类型的事件没有问题。然后我陷入了实施SomethingCreated事件的困境。
当我创建一个映射到具有身份主键的表的实体时,在实体被持久化之前我不知道Id。实体是持久性无知的,所以当从实体内部发布事件时,Id只是不知道 - 它只是在调用context.SaveChanges()之后神奇地设置。那么我如何/何时/何时将Id放入事件数据中?
我在考虑:
where id = 'B85E62C3-DC56-40C0-852A-49F759AC68FB'
,否MIN
,MAX
等。这就是我在许多示例应用程序中看到的内容。答案 0 :(得分:10)
我个人喜欢GUID的唯一标识符,特别是在数字ID导致问题的多用户分布式环境中。因此,我从不使用数据库生成的标识列/属性,这个问题就消失了。
除此之外,由于您正在关注CQRS,因此您无疑拥有CreateSomethingCommand和相应的CreateSomethingCommandHandler,它们实际执行创建新实例所需的步骤并使用存储库(通过context.SaveChanges)保留新对象。我将在这里举起SomethingCreated事件,而不是域对象本身。
首先,这解决了您的问题,因为命令处理程序可以等待数据库操作完成,提取标识值,更新对象然后在事件中传递标识。但是,更重要的是,它还解决了关于“创建”对象的具体时间的棘手问题?
在构造函数中引发域事件是不好的做法,因为构造函数应该是精简的并且只是执行初始化。另外,在您的模型中,在分配ID之前,不会真正创建对象。这意味着在执行构造函数之后还需要执行其他初始化步骤。如果你有多个步骤,你是否强制执行执行顺序(另一个反模式)或者检查每个步骤以确认它们何时完成(哦,臭)?希望你能看到它如何迅速失控。
所以,我的建议是从命令处理程序中引发事件。 (注意:即使你切换到GUID标识符,我也会遵循这种方法,因为你永远不应该从构造函数中引发事件。)