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ですが、それは問題ではありません)。この例のテーブルは、オブジェクト モデルとほぼ 1 対 1 です。各下位オブジェクトへの外部キー関係を表すクラスの Items プロパティ。つまり、3 つのテーブルで、A は B と 1 対多の関係を持ち、B は C と 1 対多の関係を持ちます。

そのため、AI の特定の ID に対して、オブジェクトにすべての子データも含める必要があります。

私の最善の推測は、何らかの形で QueryMultiple を使用する必要があるということですが、それを行う最善の方法がわかりません。

4

1 に答える 1

12

ここで提案するヘルパー:オブジェクト階層を作成する 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;
}

このパターンを拡張して、3 レベルの階層で機能させることができます。

于 2011-06-21T12:04:37.620 に答える