0

SecureRandom 需要多少次迭代才能生成给定范围之间的所有数字。例如,我正在生成 0-9(包括两者)之间的随机数。然后经过多少次迭代,所有数字(0、1、2、...、9)至少出现一次。

4

3 回答 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 的话:这就是随机性的问题,你永远无法确定 :)

http://dilbert.com/fast/2001-10-25/

于 2012-10-09T19:34:37.563 回答