45

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

4

7 に答える 7

106

mquander の回答と Dan Blanchard のコメントに加えて、 Fisher-Yates-Durstenfeld shuffleを実行する 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("source");
        if (rng == null) throw new ArgumentNullException("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];
        }
    }
}
于 2009-10-31T02:27:19.187 に答える
41

もう 1 つのオプションは、OrderBy を使用し、GUID 値で並べ替えることです。これは、次を使用して行うことができます。

var result = sequence.OrderBy(elem => Guid.NewGuid());

上記が実際にランダムな分布を生成することを確信するために、いくつかの経験的テストを行いました(そうであるように見えます)。私の結果はTechniques for Randomly Reordering an Arrayで見ることができます。

于 2010-07-02T21:19:52.843 に答える
16

これには「ランダムバイアス」に関するいくつかの問題があり、最適ではないと確信しています。これは別の可能性です。

var r = new Random();
l.OrderBy(x => r.NextDouble()).Take(n);
于 2009-10-31T18:01:15.463 に答える
6

コレクションをランダムな順序にシャッフルnし、結果から最初の項目を取得します。

于 2009-10-30T18:49:43.900 に答える
-1

私はこのオーバーライドメソッドを書きます:

public static IEnumerable<T> Randomize<T>(this IEnumerable<T> items) where T : class
{
     int max = items.Count();
     var secuencia = Enumerable.Range(1, max).OrderBy(n => n * n * (new Random()).Next());

     return ListOrder<T>(items, secuencia.ToArray());
}

private static IEnumerable<T> ListOrder<T>(IEnumerable<T> items, int[] secuencia) where T : class
        {
            List<T> newList = new List<T>();
            int count = 0;
            foreach (var seed in count > 0 ? secuencia.Skip(1) : secuencia.Skip(0))
            {
                newList.Add(items.ElementAt(seed - 1));
                count++;
            }
            return newList.AsEnumerable<T>();
        }

次に、ソースリスト(すべてのアイテム)があります

var listSource = p.Session.QueryOver<Listado>(() => pl)
                        .Where(...);

最後に、「ランダム化」を呼び出して、項目のランダムなサブコレクションを取得します。私の場合は 5 つの項目です。

var SubCollection = Randomize(listSource.List()).Take(5).ToList();
于 2014-06-10T15:29:26.183 に答える
-3

醜いコードでごめんなさい:-)、でも


var result =yourCollection.OrderBy(p => (p.GetHashCode().ToString() + Guid.NewGuid().ToString()).GetHashCode()).Take(n);

于 2009-10-31T13:39:41.197 に答える