【发布时间】:2014-01-31 13:00:39
【问题描述】:
我对如何设计聚合有疑问。
我有 Company、City、Province 和 Country 实体。这些中的每一个都需要是其自身聚合的聚合根。 City、Province 和Country 实体在整个系统中使用,并被许多其他实体引用,因此它们不是值对象,也需要在许多不同的场景中访问。所以他们应该有存储库。 CityRepository 将具有 FindById(int)、GetAll()、GetByProvince(Province)、GetByCountry(Country)、GetByName(string) 等方法。
举个例子。一个Company 实体与一个City 相关联,后者属于一个Province,而Province 又属于一个Country:
现在假设我们有一个公司列表页面,其中列出了一些公司及其所在城市、省和国家/地区。
ID 引用
如果一个实体需要引用 City、Province 或 Country,他们会通过 ID 这样做(正如 Vaughn Vernon 所建议的那样)。
为了从存储库中获取这些数据,我们需要调用 4 个不同的存储库,然后匹配数据以填充视图。
var companies = CompanyRepository.GetBySomeCriteria();
var cities = CityRepository.GetByIds(companies.Select(x => x.CityId);
var provinces = ProvinceRepository.GetByIds(cities.Select(x => x.ProvinceId);
var countries = CountryRepository.GetByIds(province.Select(x => x.CountryId);
foreach(var company in companies)
{
var city = cities.Single(x => x.CityId == company.CityId);
var province = provinces.Single(x => x.ProvinceId == city.ProvinceId);
var country = countries.Single(x => x.CountryId == province.CountryId);
someViewModel = new CompanyLineViewModel(company.Name, city.Name, province.Name, country.Name);
}
这是一种非常庞大且效率低下的方法,但显然是“正确”的方式?
引用引用
如果实体被引用引用,相同的查询将如下所示:
var companies = CompanyRepository.GetBySomeCriteria();
someViewModel = new CompanyLineViewModel(company.Name, company.City.Name, company.Province.Name, company.Country.Name);
但据我了解,这些实体不能通过引用来引用,因为它们存在于不同的聚合中。
问题
我还能如何更好地设计这些聚合?
我是否可以使用城市模型加载公司实体,即使它们存在于不同的聚合中?我想这将很快打破聚合之间的界限。在更新聚合时处理事务一致性时也会造成混乱。
【问题讨论】:
-
存储库用于聚合存储,而不是用于复杂读取。 jefclaes.be/2014/01/repositories-where-did-we-go-wrong_26.html
标签: domain-driven-design repository-pattern unit-of-work aggregateroot onion-architecture