SecureRandom 需要多少次迭代才能生成给定范围之间的所有数字。例如,我正在生成 0-9(包括两者)之间的随机数。然后经过多少次迭代,所有数字(0、1、2、...、9)至少出现一次。
问问题
795 次
3 回答
3
假设 SecureRandom 是完全随机的(它的 Secure 部分所追求的)答案是,没有给定的数字可以保证所有数字至少出现一次。
于 2012-10-09T19:28:23.027 回答
3
如前所述,无法保证产生这十个值需要多长时间。然而,概率定律确实使得它更可能需要接近 10 次迭代而不是 1,000,000 次。
我不够聪明,无法计算出实际的概率。但是,可以编写一个程序通过反复运行问题来对空间进行采样,计算每次迭代的次数,然后您可以确定,例如,50% 的时间将花费少于 n 次迭代.
只是为了好玩,我写了一些东西来尝试这个问题,将问题运行 1,000,000 次并汇总结果。运行 3 次后,我的结果显示:
- 50% 的时间,它将花费少于 27 次(包括)迭代(全部 3 次运行)。
- 90% 的时间,它将花费少于 44 次(包括)迭代(全部 3 次运行)。
- 99% 的时间,它将花费少于 66 次(包括)迭代(全部 3 次运行)。
- 99.9% 的时间,它将花费少于 88 次(包括)迭代(所有 3 次运行)。
- 每种情况下的最差运行情况不同,但 3 个值是 154 次迭代、156 次迭代和 170 次迭代。
这是我的检查代码。它使用 java.security.SecureRandom,但还包括一种使用 java.util.Random 的方法。
import java.security.SecureRandom;
import java.util.Map;
import java.util.Random;
import java.util.TreeMap;
public class HowLong {
public static final int MAX_TO_GENERATE = 10;
public static final int TOTAL_RUNS = 1000000;
public static final int TP50 = (int)(TOTAL_RUNS * 0.50);
public static final int TP90 = (int)(TOTAL_RUNS * 0.90);
public static final int TP99 = (int)(TOTAL_RUNS * 0.99);
public static final int TP99_9 = (int)(TOTAL_RUNS * 0.999);
public static final int TP100 = (int)(TOTAL_RUNS * 1);
public static final String[] TP_NAMES = {"TP50", "TP90", "TP99", "TP99.9", "TP100"};
public static final int[] TPS = { TP50, TP90, TP99, TP99_9, TP100 };
public interface RandomSource {
int next();
}
public static class MathRandomSource implements RandomSource {
private final Random rand;
public MathRandomSource() {
rand = new Random();
}
public int next() {
return rand.nextInt(MAX_TO_GENERATE);
}
}
public static class SecureRandomSource implements RandomSource {
private final SecureRandom rand;
public SecureRandomSource() {
rand = new SecureRandom();
}
public int next() {
return rand.nextInt(MAX_TO_GENERATE);
}
}
public static int waitForTen() {
final boolean[] found = new boolean[MAX_TO_GENERATE];
int remaining = found.length;
final RandomSource source = new SecureRandomSource();
int iterations = 0;
while (remaining > 0) {
int next = source.next();
if (!found[next]) {
found[next] = true;
remaining -= 1;
}
iterations += 1;
}
return iterations;
}
public static void main(String[] args) {
System.out.println("Attempting to generate all numbers below: " + MAX_TO_GENERATE);
System.out.println("Performing n iterations: " + TOTAL_RUNS);
TreeMap<Integer, Integer> results = new TreeMap<Integer, Integer>();
for (int i = 0; i < TOTAL_RUNS; i += 1) {
Integer iterations = waitForTen();
Integer currentCount = results.get(iterations);
if (currentCount == null) {
results.put(iterations, 1);
} else {
results.put(iterations, currentCount + 1);
}
}
int currTP = 0;
int count = 0;
for (Map.Entry<Integer, Integer> entry: results.entrySet()) {
count += entry.getValue();
while (currTP < TPS.length && count >= TPS[currTP]) {
System.out.println(TP_NAMES[currTP] + ": " + entry.getKey());
currTP += 1;
}
}
}
}
于 2012-10-09T20:02:50.737 回答
2
正如 CrazyCasta 所说,正确答案是 - 没有给定的数字。
引用 Scott Adams 的话:这就是随机性的问题,你永远无法确定 :)
于 2012-10-09T19:34:37.563 回答