ランダムなサブコレクションを取得するための最適な LINQ クエリ - Shuffle Ask Question

ランダムなサブコレクションを取得するための最適な LINQ クエリ - Shuffle Ask Question

'N' 個のアイテムを持つコレクションから、'n' 個のランダムにシャッフルされたコレクションを取得する最も簡単な方法を提案してください。ここで、n <= N

ベストアンサー1

mquanderの回答とDan Blanchardのコメントに加えて、以下はLINQに適した拡張メソッドです。フィッシャー・イェーツ・ダーステンフェルド・シャッフル:

// take n random items from yourCollection
var randomItems = yourCollection.Shuffle().Take(n);

// ...

public static class EnumerableExtensions
{
    public static IEnumerable<T> Shuffle<T>(this IEnumerable<T> source)
    {
        return source.Shuffle(new Random());
    }

    public static IEnumerable<T> Shuffle<T>(this IEnumerable<T> source, Random rng)
    {
        if (source == null) throw new ArgumentNullException(nameof(source));
        if (rng == null) throw new ArgumentNullException(nameof(rng));

        return source.ShuffleIterator(rng);
    }

    private static IEnumerable<T> ShuffleIterator<T>(
        this IEnumerable<T> source, Random rng)
    {
        var buffer = source.ToList();
        for (int i = 0; i < buffer.Count; i++)
        {
            int j = rng.Next(i, buffer.Count);
            yield return buffer[j];

            buffer[j] = buffer[i];
        }
    }
}

おすすめ記事