DDD:使用两个聚合根建模简单域

时间:2012-06-16 18:04:41

标签: domain-driven-design root aggregate

假设我想创建一个行动网站,其中成员可以为项目出价。为了对这个域进行建模,我有三个类:Member,Item和Bid。 我的头脑风暴会是这样的:

  • 项目可以包含多个出价
  • 出价与一个项目和一个成员相关联
  • 会员可以包含多个出价
  • 会员和项目可以不带出价实例
  • 如果没有会员和项目
  • ,则出价实例不能存在

考虑到这一切很明显,由于Member和Item对象是独立的,我们可以将它们视为聚合根。出价将是其中一个汇总的一部分。这很清楚但是现在令我困惑的是我应该选择哪个聚合根?物品或会员?

这是Apress的Pro ASP.NET MVC 3 Framework一书的例子,他们的做法如下:

enter image description here

其中提供以下代码:

public class Member 
{
    public string LoginName { get; set; } // The unique key
    public int ReputationPoints { get; set; }
}

public class Item 
{
    public int ItemID { get; private set; } // The unique key
    public string Title { get; set; }
    public string Description { get; set; }
    public DateTime AuctionEndDate { get; set; }
    public IList<Bid> Bids { get; set; }
}

public class Bid 
{
    public Member Member { get; set; }
    public DateTime DatePlaced { get; set; }
    public decimal BidAmount { get; set; }
}

会员和项目是此处的汇总根,并且Bid包含在项目中。 现在让我们说我有应用程序用例:“获取特定成员发布的所有出价”。这是否意味着我必须首先获取所有项目(例如,通过存储库接口从数据库中获取),然后枚举每个项目的所有出价,试图找到匹配的成员?这不是有点低效吗?那么更好的方法是在Member中聚合Bid对象。但在这种情况下,请考虑新的用例:“获取特定项目的所有出价”。现在我们再次需要采取其他方式来获得所有出价......

因此,考虑到我需要在我的应用程序中实现这两个用例,那么对这个域进行建模的正确有效方法是什么?

1 个答案:

答案 0 :(得分:4)

您的域名应该只反映Command(CQRS)要求(更新/更改数据)。我认为您需要查询(读取数据,不更新/更改数据):“获取特定项目的所有出价”和“获取特定成员发布的所有出价”。因此,这种“查询”与域无关,因为查询实现独立于命令实现(命令调用域方法)。这使您可以自由地以有效的方式实现每个查询。我的方法是实现一个高效的数据库视图,只获取您想要在UI中显示的数据。然后你创建一个名为BidForItemDto(DTO = data transfer object)的新类,并将数据从DB视图映射到BidForItemDto的集合(你可以通过ADO.NET手动完成或使用NHibernate(首选,为你做所有事情) )。对于第二个查询,同样创建一个名为BidPostedByMemberDto的新类。

因此,如果它是您需要的查询,只需忘记域,意识到它只是您想要在UI中显示的数据,并从数据库中有效地查询它们。只有当您在UI中执行某些操作时(单击按钮才能进行出价),它才会执行命令“出价”,这将在最后调用域方法Item.PlaceBid(成员,日期时间日期,小数金额) )。顺便说一句,恕我直言,它是一个“有很多出价”的项目,域名方法“地方出价”肯定需要访问以前的出价来正确实现整个逻辑。将出价收集到会员中对我来说没有多大意义......

从头顶看一些DB视图和SQL查询的例子:

获取特定商品的所有出价:

create view BidForItemDto
as
select 
    i.ItemId,
    b.BidId,
    b.MemberId,
    b.DatePlaced,
    b.BidAmount     
from Item i
join Bid b ON b.ItemId = i.ItemId

查询:

SELECT *
from BidFormItemDto
where ItemId = <provide item id>

获取特定成员发布的所有出价:

create view BidPostedByMemberDto
as
select 
    m.MemberId,
    b.BidId,
    b.MemberId,
    b.DatePlaced,
    b.BidAmount     
from Member m
join Bid b ON b.MemberId = i.MemberId

查询:

SELECT *
from BidPostedByMemberDto
where MemberId = <provide member id>