Шаблон репозитория без ORM
Я использую шаблон репозитория в приложении.NET C#, которое не использует ORM. Однако проблема, с которой я сталкиваюсь, заключается в том, как заполнить свойства списка "один ко многим" сущности. например, если у клиента есть список заказов, т.е. если у класса Customer есть свойство List с именем Orders, а в моем хранилище есть метод GetCustomerById, тогда?
- Нужно ли загружать список заказов в методе GetCustomerById?
- Что если у самого ордера есть другое свойство списка и так далее?
- Что делать, если я хочу сделать ленивую загрузку? Где бы я поместил код для загрузки свойства Orders в клиент? В свойстве Orders есть метод доступа {}? Но тогда я должен был бы внедрить хранилище в сущность домена? который я не считаю правильным решением.
Это также вызывает вопросы о таких функциях, как отслеживание изменений, удаление и т. Д.? Так что я думаю, что конечный результат - я могу сделать DDD без ORM?
Но сейчас меня интересует только отложенная загрузка свойств List в доменных сущностях? Любая идея?
Набиль
Я предполагаю, что это очень распространенная проблема для тех, кто не использует ORM в дизайне, управляемом доменом? Любая идея?
3 ответа
я могу сделать DDD без ORM?
Да, но ORM упрощает вещи.
Если честно, я думаю, что ваша проблема не в том, нужен ли вам ORM или нет, а в том, что вы слишком много думаете о данных, а не о поведении, что является ключом к успеху с DDD. С точки зрения модели данных, большинство сущностей будут иметь ассоциации с большинством других сущностей в той или иной форме, и с этой точки зрения вы можете обойти всю модель. Вот как это выглядит с вашим клиентом и заказами и, возможно, поэтому вы думаете, что вам нужна ленивая загрузка. Но вам нужно использовать агрегаты, чтобы разбить эти отношения на поведенческие группы.
Например, почему вы смоделировали совокупность клиентов, чтобы иметь список заказов? Если ответ "потому что у клиента могут быть заказы", то я не уверен, что вы в курсе DDD.
Какое поведение требует от клиента наличия списка заказов? Когда вы больше задумываетесь о поведении вашего домена (то есть какие данные требуются и в какой момент), вы можете смоделировать свои агрегаты на основе вариантов использования, и все станет намного понятнее и намного проще, так как вы отслеживаете изменения только для небольшого набора объектов. в совокупной границе.
Я подозреваю, что Клиент должен быть отдельным агрегатом без списка заказов, а Заказ должен быть агрегатом со списком строк заказов. Если вам необходимо выполнить операции с каждым заказом для клиента, используйте orderRepository.GetOrdersForCustomer(customerID); внесите свои изменения, затем используйте orderRespository.Save(order);
Что касается отслеживания изменений без ORM, есть несколько способов сделать это, например, агрегат заказов может вызвать события, которые репозиторий заказов прослушивает для удаленных строк заказа. Затем они могут быть удалены, когда единица работы завершена. Или чуть менее элегантный способ - поддерживать удаленные списки, т.е. order.DeletedOrderLines, которые, очевидно, может прочитать ваш репозиторий.
Чтобы подвести итог:
- Я думаю, что вам нужно больше думать о поведении, чем о данных
- ORM облегчает отслеживание изменений, но вы можете сделать это без него, и вы определенно можете сделать DDD без него.
РЕДАКТИРОВАТЬ в ответ на комментарий:
Я не думаю, что я бы реализовал ленивую загрузку для строк заказа. Какие операции вы можете выполнить с заказом без необходимости в строках заказа? Не много я подозреваю.
Однако я не ограничиваюсь "правилами" DDD, когда это кажется бессмысленным, поэтому... Если в маловероятном сценарии существует ряд операций, выполненных над объектом заказа, который не требуется заполнять строки заказа, и с заказом часто связано большое количество строк заказа (и то, и другое должно быть правдой, чтобы я считал это проблемой), тогда я бы сделал это:
Имейте это личное поле в объекте заказа:
private Func<Guid, IList<OrderLine>> _lazilyGetOrderLines;
Который будет передаваться хранилищем заказов в заказ на создание:
Order order = new Order(this.GetOrderLines);
Где это приватный метод в OrderRepository:
private IList<OrderLine> GetOrderLines(Guid orderId)
{
//DAL Code here
}
Тогда в строке порядка свойство может выглядеть так:
public IEnumberable<OrderLine> OrderLines
{
get
{
if (_orderLines == null)
_orderLines = _lazilyGetOrderLines(this.OrderId);
return _orderLines;
}
}
Редактировать 2
Я нашел этот пост в блоге, который имеет аналогичное решение, но немного более элегантный:
http://thinkbeforecoding.com/post/2009/02/07/Lazy-load-and-persistence-ignorance
1) Нужно ли загружать список заказов в методе GetCustomerById?
Вероятно, это хорошая идея, чтобы отделить код отображения заказа от кода отображения клиента. Если вы пишете свой код доступа к данным вручную, вызовите этот модуль отображения из GetCustomerById
Метод ваш лучший вариант.
2) Что если у самого Ордера есть другое свойство списка и так далее?
Логика собрать всех вместе должна где-то жить; соответствующий совокупный репозиторий так же хорош, как и любой другой.
3) Что делать, если я хочу сделать ленивую загрузку? Где бы я поместил код для загрузки свойства Orders в клиент? В свойстве Orders есть метод доступа {}? Но тогда я должен был бы внедрить хранилище в сущность домена? который я не считаю правильным решением.
Лучшее решение, которое я видел, - это заставить ваш репозиторий возвращать подклассовые доменные сущности (используя что-то вроде Castle DynamicProxy), что позволяет вам сохранять постоянное невежество в вашей доменной модели.
Другой возможный ответ - создать новый объект Proxy, который наследуется от Customer, назовите его CustomerProxy и обработать там ленивую загрузку. Все это псевдокод, так что это дает вам представление, а не просто копирует и вставляет его для использования.
Пример:
public class Customer
{
public id {get; set;}
public name {get; set;}
etc...
public virtual IList<Order> Orders {get; protected set;}
}
вот класс Customer "proxy"... этот класс находится не на бизнес-уровне, а на уровне данных вместе с вашим Context и Data Mappers. Обратите внимание, что любые коллекции, которые вы хотите сделать lazy-load, вы должны объявить как виртуальные (я полагаю, что EF 4.0 также требует, чтобы вы делали виртуальные реквизиты, как если бы прокручивали прокси-классы во время выполнения на чистых POCO, чтобы Context мог отслеживать изменения)
internal sealed class CustomerProxy : Customer
{
private bool _ordersLoaded = false;
public override IList<Order> Orders
{
get
{
IList<Order> orders = new List<Order>();
if (!_ordersLoaded)
{
//assuming you are using mappers to translate entities to db and back
//mappers also live in the data layer
CustomerDataMapper mapper = new CustomerDataMapper();
orders = mapper.GetOrdersByCustomerID(this.ID);
_ordersLoaded = true;
// Cache Cases for later use of the instance
base.Orders = orders;
}
else
{
orders = base.Orders;
}
return orders;
}
}
}
Таким образом, в этом случае наш объект-сущность Customer по-прежнему свободен от вызовов кода базы данных /datamapper, а это то, что нам нужно... "чистые" POCO. Вы делегировали отложенную загрузку прокси-объекту, который находится на уровне данных, и создает экземпляры картографических данных и выполняете вызовы.
у этого подхода есть один недостаток: вызов клиентского кода не может переопределить ленивую загрузку... он либо включен, либо выключен. Так что это зависит от вас в ваших конкретных условиях использования. Если вы знаете, что, возможно, в 75% случаев вам всегда будут нужны Заказы Клиента, ленивая нагрузка, вероятно, не лучшая ставка. Для CustomerDataMapper было бы лучше заполнить эту коллекцию в тот момент, когда вы получаете сущность Customer.
Опять же, я думаю, что и NHibernate, и EF 4.0 позволяют вам изменять характеристики отложенной загрузки во время выполнения, поэтому, как обычно, имеет смысл использовать ORM, так как большая часть функциональности предоставлена для вас.
Если вы не используете Заказы так часто, используйте ленивую загрузку для заполнения коллекции Заказов.
Я надеюсь, что это "правильно" и является способом ленивой загрузки правильного способа для моделей доменной модели. Я все еще новичок в этом деле...
Майк