如何使用Dapper高效地选择聚合对象?

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之间也有一对多的关系。

因此,对于给定的A ID,我希望我的对象也具有其所有子数据。

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


也许是 QueryMultiple 方法?我刚刚下载了 Dapper..假设你指的是 'dapper-dot-net'。 - IAbstract
是的,我正在尝试使用Dapper-dot-net。 - Peter
我目前正在做的是对所有不同表进行QueryMultiple,然后使用linq按id分配到子集合中。这意味着我只需要向数据库发送一批数据并在内存中完成所有工作,而不是像我看到的linq2sql那样向数据库发送大量的附带查询...我对结果感到非常满意,只是想知道我的方法是否符合dapper作者的意图。 - Peter
目前我们没有运行身份管理器,所以您会得到重复的内容。我需要尝试一下来解决这个问题。 - Marc Gravell
谢谢@Marc,我期待着看到你的成果。目前我对手动处理方式感到相当满意,但身份管理将是一个额外的好处。 - Peter
@Peter 我一直在思考这个问题,但是没有想到一个我喜欢的优雅的API,你有什么建议吗? - Sam Saffron
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;
}

您可以将此模式扩展为适用于三级层次结构。


1
这对我不起作用。当它尝试读取TSecond时,读取器已被消耗,无法再次读取。 - Mr Grok
谢谢Sam,我通过修改查询语句成功了。使用了你在这篇帖子中建议的类似方法 SO help with multi-mapper - BZink

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接