Linq-to-SQL ToDictionary()


81

Linqを使用してSQL(2008)から2つの列を(キャッシュ用の)辞書に適切に変換するにはどうすればよいですか?

現在、IQueryable b / cをループしていますが、ToDictionaryメソッドを機能させることができません。何か案は?これは機能します:

var query = from p in db.Table
            select p;

Dictionary<string, string> dic = new Dictionary<string, string>();

foreach (var p in query)
{
    dic.Add(sub.Key, sub.Value);
}

私が本当にやりたいのは、このようなものですが、機能していないようです。

var dic = (from p in db.Table
             select new {p.Key, p.Value })
            .ToDictionary<string, string>(p => p.Key);

しかし、このエラーが発生します:「System.Linq.IQueryable」から「System.Collections.Generic.IEnumerable」に変換できません

回答:


121
var dictionary = db
    .Table
    .Select(p => new { p.Key, p.Value })
    .AsEnumerable()
    .ToDictionary(kvp => kvp.Key, kvp => kvp.Value)
;

申し訳ありませんが、はっきりしていなかったかもしれません。以前に試したことがありますが、これにより、Dictionary <string、string>ではなくDictionary <string、#Anonymous Type>が作成されます
Codewerks 2008年

試してみてくださいkvp => kvp.Value as string。答えのポイントはでした.AsEnumerable()
yfeldblum 2008年

おかげで、ええ、AsEnumerableが必要だと思いましたが、ToDictionary呼び出しはまだ機能しません。....彼らは文字列として戻ってくるように、両方の列は、SQLでのvarcharですが、私はディックにものにそれを取得する方法を見つけ出すことはできません
Codewerks

これがまだ関連しているかどうかはわかりませんが、なぜAsEnumerable必要なのですか?私にとってはそれがなくても機能しますが、おそらくLinqはすでに変更されています(7年が経過しています)
Alexander Derck 2015年

1
@ AlexanderDerck-当時、AsEnumerableが必要でした。理由は正確には覚えていませんが、考えられる理由ToDictionaryは、の拡張メソッドでしたIEnumerableが、Linq-to-SQLのインターフェイスはそうではなかったということIEnumerableです。
yfeldblum 2015年

16

キーを定義するだけですが、値も含める必要があります。

var dic = (from p in db.Table
             select new {p.Key, p.Value })
            .ToDictionary(p => p.Key, p=> p.Value);

3
-1エラーは、<string, string>パーツがpublic static Dictionary<TKey, TSource> ToDictionary<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer);オーバーロードの代わりにオーバーロードを使用するようにすることpublic static Dictionary<TKey, TElement> ToDictionary<TSource, TKey, TElement>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector);です。
user247702 2012年

.AsEnumerable()以前はまだ必要だと思います.ToDictionary()
AceMark 2014年

完璧に機能しました...ついに私が探していたもの
ブライアン

9

みんなありがとう、あなたの答えは私がこれを修正するのを助けました、あるべきです:

var dic = db
        .Table
        .Select(p => new { p.Key, p.Value })
        .AsEnumerable()
        .ToDictionary(k=> k.Key, v => v.Value);

5
AsEnumerable()が必要な理由は、LINQ to SQLがローカル処理とリモート(SQL)処理を混合しないため、最初の部分がSQLサーバーで実行され、最後の部分がLINQ toObjectsを使用してローカルで実行されるためです。辞書を作成する:)
DamienG 2008年

理にかなっています。また、ToDictionaryの2番目のパラメーターには、以前に私をつまずかせていた独自のFunc引数が必要です。
Codewerks 2008年

1

変換するためだけに、テーブル内のすべてのアイテムに対して匿名オブジェクトを作成するのはなぜですか?

単純に次のようなものを使用できます IDictionary<string, string> dic = db.Table.ToDictionary(row => row.Key, row => row.Value); 。TableとToDictionary()の間にAsEnumerable()呼び出しを含める必要がある場合があります。db.Tableの正確なタイプがわかりません。


また、最初のサンプルを修正してください。2番目のループ変数は宣言と使用時に不一致です。


2
そのためToDictionaryに、メモリです。匿名オブジェクトを作成しないということは、必要な列だけでなく、すべての列がデータベースから取得されることを意味します。
user247702 2012年

ToDictionary基になる構造についてはわかりません。キーと値の2つのコールバック(この場合はラムダを選択する単純なプロパティ)を呼び出すだけなので、このToDictionary呼び出しは、彼が提供した例のforeachと機能的に同等である必要があります。私の知る限り、すべてのLINQメソッドは延期されます。つまり、コールバックは必要な場合にのみ呼び出されます。
TWiStErRob 2012年

3
SQL Server Profilerを使用すると、匿名オブジェクトでは2つの列のみが選択され、それがない場合はすべての列が選択されていることがわかります。
user247702 2012年
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.