オブジェクト階層を作成するためのマルチマッパー


82

文書化された投稿/ユーザーの例に非常に似ているように見えるので、私はこれを少し遊んでいますが、少し異なり、私には機能していません。

次の簡略化されたセットアップ(連絡先に複数の電話番号がある)を想定します。

public class Contact
{
    public int ContactID { get; set; }
    public string ContactName { get; set; }
    public IEnumerable<Phone> Phones { get; set; }
}

public class Phone
{
    public int PhoneId { get; set; }
    public int ContactID { get; set; } // foreign key
    public string Number { get; set; }
    public string Type { get; set; }
    public bool IsActive { get; set; }
}

複数のPhoneオブジェクトとの連絡先を返すものになりたいです。そうすれば、2つの連絡先があり、それぞれに2つの電話がある場合、SQLは、合計4行の結果セットとしてそれらの結合を返します。次に、Dapperは、それぞれ2つの電話で2つの連絡先オブジェクトをポップアウトします。

ストアドプロシージャのSQLは次のとおりです。

SELECT *
FROM Contacts
    LEFT OUTER JOIN Phones ON Phones.ReferenceId=Contacts.ReferenceId
WHERE clientid=1

これを試しましたが、4タプルになりました(これは問題ありませんが、期待していたものではありません...それでも結果を再正規化する必要があることを意味します):

var x = cn.Query<Contact, Phone, Tuple<Contact, Phone>>("sproc_Contacts_SelectByClient",
                              (co, ph) => Tuple.Create(co, ph), 
                                          splitOn: "PhoneId", param: p, 
                                          commandType: CommandType.StoredProcedure);

また、別のメソッド(以下)を試してみると、「タイプ 'System.Int32'のオブジェクトをタイプ 'System.Collections.Generic.IEnumerable`1 [Phone]'にキャストできません」という例外が発生します。

var x = cn.Query<Contact, IEnumerable<Phone>, Contact>("sproc_Contacts_SelectByClient",
                               (co, ph) => { co.Phones = ph; return co; }, 
                                             splitOn: "PhoneId", param: p,
                                             commandType: CommandType.StoredProcedure);

私は何か間違ったことをしているだけですか?投稿/所有者の例と同じように見えますが、子から親ではなく、親から子に移動する点が異なります。

前もって感謝します

回答:


69

あなたは何も悪いことをしていません、それはAPIが設計された方法ではありません。すべてのQueryAPIは、常にデータベース行ごとにオブジェクトを返します。

したがって、これは多方向->一方向ではうまく機能しますが、一方向->多マップではうまく機能しません。

ここには2つの問題があります。

  1. クエリで機能する組み込みのマッパーを導入すると、重複するデータを「破棄」することが期待されます。(Contacts。*はクエリで重複しています)

  2. 1つ->多数のペアで機能するように設計する場合、ある種のIDマップが必要になります。これにより複雑さが増します。


たとえば、限られた数のレコードをプルする必要がある場合に効率的なこのクエリを考えてみましょう。これを100万までプッシュすると、トリッキーになり、ストリーミングする必要があり、すべてをメモリにロードできなくなります。

var sql = "set nocount on
DECLARE @t TABLE(ContactID int,  ContactName nvarchar(100))
INSERT @t
SELECT *
FROM Contacts
WHERE clientid=1
set nocount off 
SELECT * FROM @t 
SELECT * FROM Phone where ContactId in (select t.ContactId from @t t)"

あなたができることはGridReader、再マッピングを可能にするためにを拡張することです:

var mapped = cnn.QueryMultiple(sql)
   .Map<Contact,Phone, int>
    (
       contact => contact.ContactID, 
       phone => phone.ContactID,
       (contact, phones) => { contact.Phones = phones };  
    );

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

これは少しトリッキーで複雑なので、注意が必要です。私はこれをコアに含めることに傾倒していません。


とてもかっこいい。これにはかなりの力があります...使い方に慣れてきたばかりだと思います。クエリのペイロードを調べて、結果セットの大きさを確認し、複数のクエリを作成してそれらを一緒にマッピングできるかどうかを確認します。
Jorin 2011年

@Jorin、他のオプションは、複数の接続を調整して結果を織り込むことです。その少しトリッキー。
サムサフラン2011年

1
また、if(childMap.TryGetvalue(..))の後にelseを追加して、子アイテムがない場合に子コレクションがデフォルトでNULLではなく空のコレクションに初期化されるようにします。このように:else {addChildren(item、new TChild [] {}); }
マリウス

1
@SamSaffron私はDapperが大好きです。ありがとうございました。でも質問があります。1対多は、SQLクエリでよく発生します。設計において、実装者が使用することを何を考えていましたか?Dapperの方法でやりたいのですが、現在はSQLの方法です。片側が通常「ドライバー」であるSQLから来るこれについてどう思いますか。なぜダッパーのメニーサイドはそうなのですか?重要なのは、オブジェクトを取得して事後に解析を行うことですか?素晴らしい図書館をありがとう。
ジョニー2017年

2
仕事に適したツールを使用していることを確認してください。大規模なデータベースパフォーマンス要件がない場合、またはシステムのベンチマークを行っていない場合は、Dapperを使用して数時間またはおそらく数日を無駄にしています。
Aluan Haddad 2017

32

参考までに-私は次のことを行うことでサムの答えを機能させました:

まず、「Extensions.cs」というクラスファイルを追加しました。2か所で「this」キーワードを「reader」に変更する必要がありました。

using System;
using System.Collections.Generic;
using System.Linq;
using Dapper;

namespace TestMySQL.Helpers
{
    public static class Extensions
    {
        public static IEnumerable<TFirst> Map<TFirst, TSecond, TKey>
            (
            this Dapper.SqlMapper.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;
        }
    }
}

次に、最後のパラメーターを変更して、次のメソッドを追加しました。

public IEnumerable<Contact> GetContactsAndPhoneNumbers()
{
    var sql = @"
SELECT * FROM Contacts WHERE clientid=1
SELECT * FROM Phone where ContactId in (select ContactId FROM Contacts WHERE clientid=1)";

    using (var connection = GetOpenConnection())
    {
        var mapped = connection.QueryMultiple(sql)    
            .Map<Contact,Phone, int>     (        
            contact => contact.ContactID,        
            phone => phone.ContactID,
            (contact, phones) => { contact.Phones = phones; }      
        ); 
        return mapped;
    }
}

24

https://www.tritac.com/blog/dappernet-by-example/をチェックしてください。 次のようなことができます。

public class Shop {
  public int? Id {get;set;}
  public string Name {get;set;}
  public string Url {get;set;}
  public IList<Account> Accounts {get;set;}
}

public class Account {
  public int? Id {get;set;}
  public string Name {get;set;}
  public string Address {get;set;}
  public string Country {get;set;}
  public int ShopId {get;set;}
}

var lookup = new Dictionary<int, Shop>()
conn.Query<Shop, Account, Shop>(@"
                  SELECT s.*, a.*
                  FROM Shop s
                  INNER JOIN Account a ON s.ShopId = a.ShopId                    
                  ", (s, a) => {
                       Shop shop;
                       if (!lookup.TryGetValue(s.Id, out shop)) {
                           lookup.Add(s.Id, shop = s);
                       }
                       shop.Accounts.Add(a);
                       return shop;
                   },
                   ).AsQueryable();
var resultList = lookup.Values;

これはdapper.netテストから取得しました:https://code.google.com/p/dapper-dot-net/source/browse/Tests/Tests.cs#1343


2
うわー!私にとって、これが最も簡単な解決策であることがわかりました。確かに、1つ->多くの場合(2つのテーブルを想定)、二重選択を使用します。しかし、私の場合、私は1つ-> 1つ->多くを持っており、これはうまく機能します。さて、それは多くの冗長データを取り戻しますが、私の場合、この冗長性は比較的小さく、せいぜい10行です。
code5 2014

これは2つのレベルでうまく機能しますが、それ以上になると注意が必要になります。
Samir Aguiar 2017

1
子データがない場合、(s、a)コードはa = nullで呼び出され、アカウントには空ではなくnullエントリのリストが含まれます。「shop.Accounts.Add(a)」の前に「if(a!= null)」を追加する必要があります
Etienne Charland

12

マルチ結果セットのサポート

あなたの場合、複数の結果セットクエリを使用する方がはるかに優れています(そして簡単です)。これは単に、2つのselectステートメントを記述する必要があることを意味します。

  1. 連絡先を返すもの
  2. そして彼らの電話番号を返すもの

このようにして、オブジェクトは一意になり、重複しなくなります。


1
他の答えは独自の方法でエレガントかもしれませんが、コードが推論しやすいので、私はこれが好きになる傾向があります。少数のselectステートメントと約30行のforeach / linqコードを使用して、数レベルの深さの階層を構築できます。これは大量の結果セットで失敗する可能性がありますが、幸いなことに私はその問題を抱えていません(まだ)。
Sam Storie 2015年

10

これは、非常に使いやすい再利用可能なソリューションです。アンドリュースの答えを少し変更したものです。

public static IEnumerable<TParent> QueryParentChild<TParent, TChild, TParentKey>(
    this IDbConnection connection,
    string sql,
    Func<TParent, TParentKey> parentKeySelector,
    Func<TParent, IList<TChild>> childSelector,
    dynamic param = null, IDbTransaction transaction = null, bool buffered = true, string splitOn = "Id", int? commandTimeout = null, CommandType? commandType = null)
{
    Dictionary<TParentKey, TParent> cache = new Dictionary<TParentKey, TParent>();

    connection.Query<TParent, TChild, TParent>(
        sql,
        (parent, child) =>
            {
                if (!cache.ContainsKey(parentKeySelector(parent)))
                {
                    cache.Add(parentKeySelector(parent), parent);
                }

                TParent cachedParent = cache[parentKeySelector(parent)];
                IList<TChild> children = childSelector(cachedParent);
                children.Add(child);
                return cachedParent;
            },
        param as object, transaction, buffered, splitOn, commandTimeout, commandType);

    return cache.Values;
}

使用例

public class Contact
{
    public int ContactID { get; set; }
    public string ContactName { get; set; }
    public List<Phone> Phones { get; set; } // must be IList

    public Contact()
    {
        this.Phones = new List<Phone>(); // POCO is responsible for instantiating child list
    }
}

public class Phone
{
    public int PhoneID { get; set; }
    public int ContactID { get; set; } // foreign key
    public string Number { get; set; }
    public string Type { get; set; }
    public bool IsActive { get; set; }
}

conn.QueryParentChild<Contact, Phone, int>(
    "SELECT * FROM Contact LEFT OUTER JOIN Phone ON Contact.ContactID = Phone.ContactID",
    contact => contact.ContactID,
    contact => contact.Phones,
    splitOn: "PhoneId");

7

Sam Saffron(およびMike Gleason)のアプローチに基づいて、複数の子と複数のレベルを可能にするソリューションを次に示します。

using System;
using System.Collections.Generic;
using System.Linq;
using Dapper;

namespace TestMySQL.Helpers
{
    public static class Extensions
    {
        public static IEnumerable<TFirst> MapChild<TFirst, TSecond, TKey>
            (
            this SqlMapper.GridReader reader,
            List<TFirst> parent,
            List<TSecond> child,
            Func<TFirst, TKey> firstKey,
            Func<TSecond, TKey> secondKey,
            Action<TFirst, IEnumerable<TSecond>> addChildren
            )
        {
            var childMap = child
                .GroupBy(secondKey)
                .ToDictionary(g => g.Key, g => g.AsEnumerable());
            foreach (var item in parent)
            {
                IEnumerable<TSecond> children;
                if (childMap.TryGetValue(firstKey(item), out children))
                {
                    addChildren(item, children);
                }
            }
            return parent;
        }
    }
}

次に、関数の外部で読み取ることができます。

using (var multi = conn.QueryMultiple(sql))
{
    var contactList = multi.Read<Contact>().ToList();
    var phoneList = multi.Read<Phone>().ToList;
    contactList = multi.MapChild
        (
            contactList,
            phoneList,
            contact => contact.Id, 
            phone => phone.ContactId,
            (contact, phone) => {contact.Phone = phone;}
        ).ToList();
    return contactList;
}

次に、同じ親オブジェクトを使用して、次の子オブジェクトに対してmap関数を再度呼び出すことができます。map関数とは別に、親または子のreadステートメントに分割を実装することもできます。

これが「シングルからN」の追加の拡張メソッドです

    public static TFirst MapChildren<TFirst, TSecond, TKey>
        (
        this SqlMapper.GridReader reader,
        TFirst parent,
        IEnumerable<TSecond> children,
        Func<TFirst, TKey> firstKey,
        Func<TSecond, TKey> secondKey,
        Action<TFirst, IEnumerable<TSecond>> addChildren
        )
    {
        if (parent == null || children == null || !children.Any())
        {
            return parent;
        }

        Dictionary<TKey, IEnumerable<TSecond>> childMap = children
            .GroupBy(secondKey)
            .ToDictionary(g => g.Key, g => g.AsEnumerable());

        if (childMap.TryGetValue(firstKey(parent), out IEnumerable<TSecond> foundChildren))
        {
            addChildren(parent, foundChildren);
        }

        return parent;
    }

2
これをありがとう-素晴らしい解決策。子がない場合にaddChilderを呼び出さない代わりに、呼び出し元の関数がnullを処理できるように、ifステートメントを削除しました。そうすれば、作業がはるかに簡単な空のリストを追加できます。
Mladen Mihajlovic 2016

1
これは素晴らしい解決策です。「動的発見」に関していくつか問題がありました。これは、このcontactList = multi.MapChild <Contact、Phone、int>(/ *上記と同じコード* /
granadaCoder 2018年

4

DataAccessLayerをストアドプロシージャに移動することを決定すると、これらのプロシージャは多くの場合、複数のリンクされた結果を返します(以下の例)。

まあ、私のアプローチはほとんど同じですが、おそらくもう少し快適です。

コードは次のようになります。

using ( var conn = GetConn() )
{
    var res = await conn
        .StoredProc<Person>( procName, procParams )
        .Include<Book>( ( p, b ) => p.Books = b.Where( x => x.PersonId == p.Id ).ToList() )
        .Include<Course>( ( p, c ) => p.Courses = c.Where( x => x.PersonId == p.Id ).ToList() )
        .Include<Course, Mark>( ( c, m ) => c.Marks = m.Where( x => x.CourseId == c.Id ).ToList() )
        .Execute();
}


分解してみましょう...

拡張:

public static class SqlExtensions
{
    public static StoredProcMapper<T> StoredProc<T>( this SqlConnection conn, string procName, object procParams )
    {
        return StoredProcMapper<T>
            .Create( conn )
            .Call( procName, procParams );
    }
}

マッパー:

public class StoredProcMapper<T>
{
    public static StoredProcMapper<T> Create( SqlConnection conn )
    {
        return new StoredProcMapper<T>( conn );
    }

    private List<MergeInfo> _merges = new List<MergeInfo>();

    public SqlConnection Connection { get; }
    public string ProcName { get; private set; }
    public object Parameters { get; private set; }

    private StoredProcMapper( SqlConnection conn )
    {
        Connection = conn;
        _merges.Add( new MergeInfo( typeof( T ) ) );
    }

    public StoredProcMapper<T> Call( object procName, object parameters )
    {
        ProcName = procName.ToString();
        Parameters = parameters;

        return this;
    }

    public StoredProcMapper<T> Include<TChild>( MergeDelegate<T, TChild> mapper )
    {
        return Include<T, TChild>( mapper );
    }

    public StoredProcMapper<T> Include<TParent, TChild>( MergeDelegate<TParent, TChild> mapper )
    {
        _merges.Add( new MergeInfo<TParent, TChild>( mapper ) );
        return this;
    }

    public async Task<List<T>> Execute()
    {
        if ( string.IsNullOrEmpty( ProcName ) )
            throw new Exception( $"Procedure name not specified! Please use '{nameof(Call)}' method before '{nameof( Execute )}'" );

        var gridReader = await Connection.QueryMultipleAsync( 
            ProcName, Parameters, commandType: CommandType.StoredProcedure );

        foreach ( var merge in _merges )
        {
            merge.Result = gridReader
                .Read( merge.Type )
                .ToList();
        }

        foreach ( var merge in _merges )
        {
            if ( merge.ParentType == null )
                continue;

            var parentMerge = _merges.FirstOrDefault( x => x.Type == merge.ParentType );

            if ( parentMerge == null )
                throw new Exception( $"Wrong parent type '{merge.ParentType.FullName}' for type '{merge.Type.FullName}'." );

            foreach ( var parent in parentMerge.Result )
            {
                merge.Merge( parent, merge.Result );
            }
        }

        return _merges
            .First()
            .Result
            .Cast<T>()
            .ToList();
    }

    private class MergeInfo
    {
        public Type Type { get; }
        public Type ParentType { get; }
        public IEnumerable Result { get; set; }

        public MergeInfo( Type type, Type parentType = null )
        {
            Type = type;
            ParentType = parentType;
        }

        public void Merge( object parent, IEnumerable children )
        {
            MergeInternal( parent, children );
        }

        public virtual void MergeInternal( object parent, IEnumerable children )
        {

        }
    }

    private class MergeInfo<TParent, TChild> : MergeInfo
    {
        public MergeDelegate<TParent, TChild> Action { get; }

        public MergeInfo( MergeDelegate<TParent, TChild> mergeAction )
            : base( typeof( TChild ), typeof( TParent ) )
        {
            Action = mergeAction;
        }

        public override void MergeInternal( object parent, IEnumerable children )
        {
            Action( (TParent)parent, children.Cast<TChild>() );
        }
    }

    public delegate void MergeDelegate<TParent, TChild>( TParent parent, IEnumerable<TChild> children );
}

それだけですが、簡単なテストを行いたい場合は、モデルと手順を以下に示します。

モデル:

public class Person
{
    public Guid Id { get; set; }
    public string Name { get; set; }

    public List<Course> Courses { get; set; }
    public List<Book> Books { get; set; }

    public override string ToString() => Name;
}

public class Book
{
    public Guid Id { get; set; }
    public Guid PersonId { get; set; }
    public string Name { get; set; }

    public override string ToString() => Name;
}

public class Course
{
    public Guid Id { get; set; }
    public Guid PersonId { get; set; }
    public string Name { get; set; }

    public List<Mark> Marks { get; set; }

    public override string ToString() => Name;
}

public class Mark
{
    public Guid Id { get; set; }
    public Guid CourseId { get; set; }
    public int Value { get; set; }

    public override string ToString() => Value.ToString();
}

SP:

if exists ( 
    select * 
    from sysobjects 
    where  
        id = object_id(N'dbo.MultiTest')
        and ObjectProperty( id, N'IsProcedure' ) = 1 )
begin
    drop procedure dbo.MultiTest
end
go

create procedure dbo.MultiTest
    @PersonId UniqueIdentifier
as
begin

    declare @tmpPersons table 
    (
        Id UniqueIdentifier,
        Name nvarchar(50)
    );

    declare @tmpBooks table 
    (
        Id UniqueIdentifier,
        PersonId UniqueIdentifier,
        Name nvarchar(50)
    )

    declare @tmpCourses table 
    (
        Id UniqueIdentifier,
        PersonId UniqueIdentifier,
        Name nvarchar(50)
    )

    declare @tmpMarks table 
    (
        Id UniqueIdentifier,
        CourseId UniqueIdentifier,
        Value int
    )

--------------------------------------------------

    insert into @tmpPersons
    values
        ( '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Иван' ),
        ( '467953a5-cb5f-4d06-9fad-505b3bba2058', N'Василий' ),
        ( '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'Алефтина' )


    insert into @tmpBooks
    values
        ( NewId(), '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Книга Математика' ),
        ( NewId(), '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Книга Физика' ),
        ( NewId(), '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Книга Геометрия' ),

        ( NewId(), '467953a5-cb5f-4d06-9fad-505b3bba2058', N'Книга Биология' ),
        ( NewId(), '467953a5-cb5f-4d06-9fad-505b3bba2058', N'Книга Химия' ),

        ( NewId(), '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'Книга История' ),
        ( NewId(), '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'Книга Литература' ),
        ( NewId(), '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'Книга Древне-шумерский диалект иврита' )


    insert into @tmpCourses
    values
        ( '30945b68-a6ef-4da8-9a35-d3b2845e7de3', '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Математика' ),
        ( '7881f090-ccd6-4fb9-a1e0-ff4ff5c18450', '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Физика' ),
        ( '92bbefd1-9fec-4dc7-bb58-986eadb105c8', '576fb8e8-41a2-43a9-8e77-a8213aa6e387', N'Геометрия' ),

        ( '923a2f0c-c5c7-4394-847c-c5028fe14711', '467953a5-cb5f-4d06-9fad-505b3bba2058', N'Биология' ),
        ( 'ace50388-eb05-4c46-82a9-5836cf0c988c', '467953a5-cb5f-4d06-9fad-505b3bba2058', N'Химия' ),

        ( '53ea69fb-6cc4-4a6f-82c2-0afbaa8cb410', '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'История' ),
        ( '7290c5f7-1000-4f44-a5f0-6a7cf8a8efab', '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'Литература' ),
        ( '73ac366d-c7c2-4480-9513-28c17967db1a', '52a719bf-6f1f-48ac-9e1f-4532cfc70d96', N'Древне-шумерский диалект иврита' )

    insert into @tmpMarks
    values
        ( NewId(), '30945b68-a6ef-4da8-9a35-d3b2845e7de3', 98 ),
        ( NewId(), '30945b68-a6ef-4da8-9a35-d3b2845e7de3', 87 ),
        ( NewId(), '30945b68-a6ef-4da8-9a35-d3b2845e7de3', 76 ),

        ( NewId(), '7881f090-ccd6-4fb9-a1e0-ff4ff5c18450', 89 ),
        ( NewId(), '7881f090-ccd6-4fb9-a1e0-ff4ff5c18450', 78 ),
        ( NewId(), '7881f090-ccd6-4fb9-a1e0-ff4ff5c18450', 67 ),

        ( NewId(), '92bbefd1-9fec-4dc7-bb58-986eadb105c8', 79 ),
        ( NewId(), '92bbefd1-9fec-4dc7-bb58-986eadb105c8', 68 ),
        ( NewId(), '92bbefd1-9fec-4dc7-bb58-986eadb105c8', 75 ),
        ----------
        ( NewId(), '923a2f0c-c5c7-4394-847c-c5028fe14711', 198 ),
        ( NewId(), '923a2f0c-c5c7-4394-847c-c5028fe14711', 187 ),
        ( NewId(), '923a2f0c-c5c7-4394-847c-c5028fe14711', 176 ),

        ( NewId(), 'ace50388-eb05-4c46-82a9-5836cf0c988c', 189 ),
        ( NewId(), 'ace50388-eb05-4c46-82a9-5836cf0c988c', 178 ),
        ( NewId(), 'ace50388-eb05-4c46-82a9-5836cf0c988c', 167 ),
        ----------
        ( NewId(), '53ea69fb-6cc4-4a6f-82c2-0afbaa8cb410', 8 ),
        ( NewId(), '53ea69fb-6cc4-4a6f-82c2-0afbaa8cb410', 7 ),
        ( NewId(), '53ea69fb-6cc4-4a6f-82c2-0afbaa8cb410', 6 ),

        ( NewId(), '7290c5f7-1000-4f44-a5f0-6a7cf8a8efab', 9 ),
        ( NewId(), '7290c5f7-1000-4f44-a5f0-6a7cf8a8efab', 8 ),
        ( NewId(), '7290c5f7-1000-4f44-a5f0-6a7cf8a8efab', 7 ),

        ( NewId(), '73ac366d-c7c2-4480-9513-28c17967db1a', 9 ),
        ( NewId(), '73ac366d-c7c2-4480-9513-28c17967db1a', 8 ),
        ( NewId(), '73ac366d-c7c2-4480-9513-28c17967db1a', 5 )

--------------------------------------------------

    select * from @tmpPersons
    select * from @tmpBooks
    select * from @tmpCourses
    select * from @tmpMarks

end
go

1
このアプローチが今のところ注目やコメントを受けていない理由はわかりませんが、非常に興味深く、論理的に構造化されていると思います。共有していただきありがとうございます。このアプローチは、テーブル値関数やSQL文字列にも適用できると思います。コマンドタイプが異なるだけです。一部の拡張機能/オーバーロードだけで、これはすべての一般的なクエリタイプで機能するはずです。
グリム

私がこの権利を読んでいることを確認するために、これはユーザーがプロシージャが結果を返すタイプの順序を正確に知っている必要があります、そうですか?たとえば、Include <Book>とInclude <Course>を入れ替えた場合、これはスローされますか?
cubesnyc

@cubesnycそれがスローされるかどうかは覚えていませんが、はい、ユーザーは順序を知っている必要があります
Sam Sch 2010

2

この問題の解決策を共有し、私が使用したアプローチについて建設的なフィードバックがあるかどうかを確認したいと思いました。

私が取り組んでいるプロジェクトには、最初に説明する必要のあるいくつかの要件があります。

  1. これらのクラスはAPIラッパーで公に共有されるため、POCOを可能な限りクリーンに保つ必要があります。
  2. 上記の要件のため、私のPOCOは別のクラスライブラリにあります
  3. データに応じて異なる複数のオブジェクト階層レベルがあります(したがって、Generic Type Mapperを使用できないか、起こりうるすべての不測の事態に対応するために大量のオブジェクト階層レベルを作成する必要があります)

したがって、私が行ったことは、次のように元の行の列として単一のJSON文字列を返すことによってSQLに2番目からn番目のレベルの階層を処理させることです説明のために他の列/プロパティなどを削除しました):

Id  AttributeJson
4   [{Id:1,Name:"ATT-NAME",Value:"ATT-VALUE-1"}]

次に、私のPOCOは次のように構築されます。

public abstract class BaseEntity
{
    [KeyAttribute]
    public int Id { get; set; }
}

public class Client : BaseEntity
{
    public List<ClientAttribute> Attributes{ get; set; }
}
public class ClientAttribute : BaseEntity
{
    public string Name { get; set; }
    public string Value { get; set; }
}

POCOがBaseEntityから継承する場所。(説明のために、クライアントオブジェクトの「属性」プロパティで示されているように、かなり単純な単一レベルの階層を選択しました。)

次に、データレイヤーに、POCOから継承する次の「データクラス」がありClientます。

internal class dataClient : Client
{
    public string AttributeJson
    {
        set
        {
            Attributes = value.FromJson<List<ClientAttribute>>();
        }
    }
}

上記のように、SQLがAttributeJsondataClientクラスのプロパティにマップされている「AttributeJson」という列を返していることがわかります。これにはAttributes、継承されたClientクラスのプロパティにJSONを逆シリアル化するセッターのみがあります。dataClientクラスはinternalデータアクセス層にあり、ClientProvider(私のデータファクトリ)は元のクライアントPOCOを次のように呼び出し元のアプリ/ライブラリに返します。

var clients = _conn.Get<dataClient>();
return clients.OfType<Client>().ToList();

Dapper.Contribを使用していて、Get<T>を返す新しいメソッドを追加したことに注意してください。IEnumerable<T>

このソリューションで注意すべき点がいくつかあります。

  1. JSONシリアル化には明らかなパフォーマンスのトレードオフがあります-これを2つのサブList<T>プロパティを持つ1050行に対してベンチマークしました。それぞれがリストに2つのエンティティを持ち、279ミリ秒でクロックインします-これは私のプロジェクトのニーズに受け入れられます-これもSQL側の最適化がゼロなので、そこで数ミリ秒を短縮できるはずです。

  2. これはList<T>、必要なプロパティごとにJSONを構築するために追加のSQLクエリが必要であることを意味しますが、SQLをよく知っていて、ダイナミクスやリフレクションなどにあまり精通していないので、これは私に適しています。私は実際に内部で何が起こっているのかを理解しているので、物事をより細かく制御できます:-)

これよりも良い解決策があるかもしれません。もしあれば、あなたの考えを聞いていただければ幸いです。これは、これまでのところこのプロジェクトのニーズに合った解決策です(ただし、これは投稿の段階では実験的なものです)。 )。


これは面白い。SQL部分を共有できる可能性はありますか?
WhiteRuski 2017年
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.