Как эффективно выбрать агрегатный объект с помощью Dapper? - PullRequest
15 голосов
/ 27 мая 2011

Допустим, у меня есть ряд объектов, которые образуют совокупность.

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.

Таким образом, для данного идентификатора AI необходимо, чтобы мои объекты также имели все свои дочерние данные.1006 *

Мое лучшее предположение, что я должен каким-то образом использовать QueryMultiple, но я не уверен, как лучше это сделать.

1 Ответ

11 голосов
/ 21 июня 2011

Я думаю, что помощник, которого я предлагаю здесь: 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;
}

Вы можете расширить этот шаблон для работы с трехуровневой иерархией.

...