mat*_*i82 2 domain-driven-design aggregate root
假设我要创建一个行动网站,会员可以在其中竞标物品。为了对此域建模,我有三个类:成员,项目和出价。我的头脑风暴将是这样的:
考虑到所有这些,很明显,由于Member和Item对象是独立的,因此我们可以认为它们是集合根。出价将是其中一项的一部分。很明显,但是现在让我感到困惑的是,我应该选择哪个聚合根?项目或会员?
这是Apress的Pro ASP.NET MVC 3 Framework书中的示例,其执行方式如下所示:

给出以下代码:
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; }
}
Run Code Online (Sandbox Code Playgroud)
成员和项目是此处的汇总根,而投标包含在项目中。现在假设我有一个应用程序用例:“获取特定成员发布的所有出价”。这是否意味着我必须先获取所有商品(例如,通过存储库接口从数据库中获取),然后枚举每个商品的所有出价,以查找匹配的会员?这不是效率低下吗?因此,更好的方法是在成员内部聚合Bid对象。但是在这种情况下,请考虑新的用例:“获取特定项目的所有出价”。现在我们再次需要采取其他方法来获得所有出价...
因此,考虑到我需要在我的应用程序中实现这两个用例,那么对该领域进行建模的正确而有效的方法是什么?
您的域实际上应该只反映Command(CQRS)要求(更新/更改数据)。我假设您需要查询(读取数据,没有数据的更新/更改):“获取特定项目的所有出价”和“获取特定成员发布的所有出价”。因此,此“查询”与域无关,因为查询实现独立于命令实现(命令正在调用域方法)。这使您可以自由地以有效的方式实现每个查询。我的方法是实现有效的数据库视图,从而仅获取要在UI中显示的数据。然后,创建一个名为BidForItemDto的新类(DTO = 数据传输对象),然后将数据库视图中的数据映射到BidForItemDto的集合中(您可以通过ADO.NET手动进行操作,也可以使用NHibernate(首选为您完成所有操作))。与第二个查询相同,创建一个名为BidPostedByMemberDto的新类。
因此,如果需要查询,则只需忽略域,意识到它只是要在UI中显示的数据,然后可以从数据库中高效地查询它们。仅当您在UI中执行某些操作(例如,单击按钮以放置出价)时,它才执行“放置出价”命令,该命令将在最终调用域方法Item.PlaceBid(Member member,DateTime date,小数位数) )。顺便说一句,恕我直言,这是一个“具有很多出价”的项目,而域方法“地方出价”肯定需要访问先前的出价才能正确实现整个逻辑。对我来说,将出价收集放入会员没有太大意义。
在我的头上,有一些数据库视图和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
Run Code Online (Sandbox Code Playgroud)
查询:
SELECT *
from BidFormItemDto
where ItemId = <provide item id>
Run Code Online (Sandbox Code Playgroud)
获取特定成员发布的所有出价:
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
Run Code Online (Sandbox Code Playgroud)
查询:
SELECT *
from BidPostedByMemberDto
where MemberId = <provide member id>
Run Code Online (Sandbox Code Playgroud)