ランダムなサブコレクションを取得するための最適なLINQクエリ-シャッフル
-
22-07-2019 - |
質問
「N」個のアイテムを持つコレクションからランダムにシャッフルされたカウント「n」のコレクションを取得する最も簡単な方法を提案してください。ここで、n <!> lt; = N
解決
別のオプションは、OrderByを使用し、GUID値でソートすることです。これは、次を使用して実行できます。
var result = sequence.OrderBy(elem => Guid.NewGuid());
私は、上記が実際にランダムな分布を生成することを確信させるために、いくつかの実証的テストを行いました(これは実行されるようです)。私の結果はランダムに配列を並べ替える方法で見ることができます。
他のヒント
mquanderの答えとDan Blanchardのコメントに加えて、 Fisher-Yates-Durstenfeldシャッフル:
// 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];
}
}
}
これには<!> quot; random bias <!> quot;の問題がいくつかあります。最適ではないと確信しています。これは別の可能性です。
var r = new Random();
l.OrderBy(x => r.NextDouble()).Take(n);
コレクションをランダムに並べ替え、最初の<結果からcode> n 個のアイテム。
少しランダムではないが、効率的:
var rnd = new Random();
var toSkip = list.Count()-n;
if (toSkip > 0)
toSkip = rnd.Next(toSkip);
else
toSkip=0;
var randomlySelectedSequence = list.Skip(toSkip).Take(n);
このオーバーライドメソッドを書く:
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();
codeいコードはごめんなさい:-)、しかし
var result =yourCollection.OrderBy(p => (p.GetHashCode().ToString() + Guid.NewGuid().ToString()).GetHashCode()).Take(n);
所属していません StackOverflow