我对如何设计聚合有疑问。
我有,Company
和City
实体。这些中的每一个都需要是其自身聚合的聚合根。和实体在整个系统中使用,并且被许多其他实体引用,因此它们不是值对象,也需要在许多不同的场景中访问。所以他们应该有存储库。A将具有诸如, , , ,之类的方法。Province
Country
City
Province
Country
CityRepository
FindById(int)
GetAll()
GetByProvince(Province)
GetByCountry(Country)
GetByName(string)
举个例子。一个Company
实体与 a 相关联City
,它属于 a ,Province
它属于 a Country
:
现在假设我们有一个公司列表页面,其中列出了一些公司及其城市、省和国家。
按 ID 引用
如果一个实体需要引用 a或City
,他们将通过 ID 这样做(如 Vaughn Vernon 所建议)。 Province
Country
为了从存储库中获取这些数据,我们需要调用 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);
但据我了解,这些实体不能通过引用来引用,因为它们存在于不同的聚合中。
问题
我还能如何更好地设计这些聚合?
即使它们存在于不同的聚合中,我是否可以使用城市模型加载公司实体?我想这将很快打破聚合之间的界限。在更新聚合时处理事务一致性时也会造成混乱。