17

假设我有一系列形成聚合的对象。

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 ID,我希望我的对象也拥有它们的所有子数据。

我最好的猜测是我应该以某种方式使用 QueryMultiple 但我不确定如何最好地做到这一点。

4

1 回答 1

12

我认为我在这里提出的帮手:Multi-Mapper to create object hierarchy可能会有所帮助。

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

您可以扩展此模式以使用 3 级层次结构。

于 2011-06-21T12:04:37.620 回答