Как эффективно выбрать совокупный объект с помощью 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.

Итак, для данного идентификатора A я хочу, чтобы мои объекты также имели все свои дочерние данные.

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


person Peter    schedule 27.05.2011    source источник
comment
Может метод QueryMultiple? Я только что загрузил dapper ... если вы имеете в виду dapper-dot-net.   -  person IAbstract    schedule 27.05.2011
comment
yep dapper-dot-net - это то, с чем я экспериментирую.   -  person Peter    schedule 27.05.2011
comment
В настоящее время я использую QueryMultiple для всех различных таблиц, а затем использую linq для назначения по идентификатору дочерним коллекциям. это означает, что я делаю один пакет для базы данных и выполняю всю работу в оперативной памяти, а не то, что я видел в linq2sql, который представляет собой сброс случайных запросов к базе данных ... Я очень доволен результатом, который я просто интересно, был ли мой подход намерением автора dapper.   -  person Peter    schedule 27.05.2011
comment
На данный момент у нас нет диспетчера удостоверений, так что вы получите дубли. Мне нужно будет сыграть, чтобы обойти это.   -  person Marc Gravell    schedule 27.05.2011
comment
Спасибо, @Marc, я с нетерпением жду ваших идей. Я очень доволен ручным способом, которым я управляю сейчас, но управление идентификацией было бы бонусом.   -  person Peter    schedule 30.05.2011
comment
@Peter Я думал об этой проблеме, но не могу придумать элегантный API, который мне нравится, какие-либо предложения?   -  person Sam Saffron    schedule 31.05.2011


Ответы (1)


Я думаю, что помощник, который я предлагаю здесь: Multi-Mapper to Создание иерархии объектов может помочь.

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;
}

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

person Sam Saffron    schedule 21.06.2011
comment
У меня это не работает. Когда он пытается прочитать TSecond, считыватель был использован и не может быть прочитан снова. - person Mr Grok; 12.07.2011
comment
Спасибо, Сэм, я заставил его работать, изменив свой запрос. Использовано что-то вроде того, что вы предложили в этом сообщении SO help с мульти-картограф - person BZink; 29.07.2011