Как эффективно выбрать агрегатный объект с помощью Dapper?
Допустим, у меня есть ряд объектов, которые образуют совокупность.
public class C{
public string Details {get;set;}
}
public class B{
public string Details {get;set;}
public List<C> Items {get;set;}
}
public class A{
public long ID {get;set;}
public string Details {get;set;}
public List<B> Items {get;set;}
}
используя Dapper, каков наилучший способ заполнить их из таблиц в базе данных (в моем случае это postgres, но это не должно иметь значения). Таблицы в примере в значительной степени один к одному с объектной моделью. Свойство Items в классе, представляющее отношения внешнего ключа к каждому подчиненному объекту. то есть 3 таблицы, A имеет отношение один ко многим с B, B имеет отношение один ко многим с C.
Поэтому для данного идентификатора ИИ нужно, чтобы мои объекты также имели все свои дочерние данные.
Мое лучшее предположение, что я должен как-то использовать QueryMultiple, но я не уверен, как лучше это сделать.
1 ответ
Я думаю, что помощник, который я предлагаю здесь: Multi-Mapper для создания иерархии объектов, может помочь.
var mapped = cnn.QueryMultiple(sql)
.Map<A,B,A>
(
A => A.ID,
B => B.AID,
a, bees => { A.Items = bees};
);
Предполагая, что вы расширяете свой GridReader и с помощью картографа:
public static IEnumerable<TFirst> Map<TFirst, TSecond, TKey>
(
this GridReader reader,
Func<TFirst, TKey> firstKey,
Func<TSecond, TKey> secondKey,
Action<TFirst, IEnumerable<TSecond>> addChildren
)
{
var first = reader.Read<TFirst>().ToList();
var childMap = reader
.Read<TSecond>()
.GroupBy(s => secondKey(s))
.ToDictionary(g => g.Key, g => g.AsEnumerable());
foreach (var item in first)
{
IEnumerable<TSecond> children;
if(childMap.TryGetValue(firstKey(item), out children))
{
addChildren(item,children);
}
}
return first;
}
Вы можете расширить этот шаблон для работы с трехуровневой иерархией.