DDD:使用EntityFramework的话,如果只为聚合根设计仓储,其它实体如何处理?

背景

DDD中只有聚合根可以有仓储,仓储负责整个聚合持久化的相关生命周期,在不使用工作单元或POCO的情况下,我们可以让Order内部直接调用DAL操作OrderItem。我们也可以让Order跟踪所有OrderItem的状态,然后在OrderRepository内部操作OrderItem。如果我们采用了重量级的ORM工具,如:EntityFramework,事情会不会变得简单呢?

使用EntityFramework持久化聚合

关键思路:双主键。

示例聚合

这里以订单和订单项为例。

Order管理OrderItem

         public void AddOrderItem(OrderItem item)
{
item.MustNotNull("item"); this.State.BeforeChangeOrder(); item.OrderId = this.Id;
this.OrderItemCollection.Add(item);
this.Total += item.Subtotal;
} public void RemoveOrderItem(Guid productId)
{
this.State.BeforeChangeOrder(); var item = this.OrderItemCollection.First(x => x.ProductId == productId);
this.OrderItemCollection.Remove(item);
this.Total -= item.Subtotal;
}

映射配置

             modelBuilder
.Entity<OrderItem>()
.HasKey(x => new { x.Id, x.OrderId }); modelBuilder
.Entity<Order>()
.HasKey(x => x.Id);
modelBuilder
.Entity<Order>()
.Property(x => x.OptimisticKey).IsRowVersion().IsConcurrencyToken();
modelBuilder
.Entity<Order>()
.HasMany(x => x.OrderItemCollection).WithRequired().HasForeignKey(x => x.OrderId).WillCascadeOnDelete();

注意:上面为OrderItem配置了双主键,如果不这么配置,删除逻辑会错误,要么删除后提交失败,要么OrderItem的数据库记录还在,只是OrderId变为NULL了。

备注

考虑到聚合内非聚合根的实体都具有“本地标识”,采用“双主键”就非常合理了,这个错误新手经常犯的。

再说一个题外话,聚合内的实体的标识能被其它聚合引用吗?我觉得应该是可以的,前提是必须同时引用其完整的标识,即:双主键。

上一篇:vim设置golang语法高亮 (Centos)


下一篇:flume自定义Source(taildirSource),自定义Sink(数据库),开发完整步骤