如何使用Java中的Math.random()以相等的概率生成随机数

时间:2013-10-19 12:23:03

标签: java random numbers equals probability

我正在进行一项练习,我需要生成一个随机数,该数字可以是4个值中的一个。 (注意我只允许使用Math.random())

0 1 2 或3

目前我使用的是:randno = (int) (Math.random()*4); // range 0-3

然而,结果必须具有相同的概率。到目前为止,我的测试(尽管方法不足)表明3的发生率远低于其他数字。

这是巧合吗?或者我的发电机没有相同的概率。

谢谢!

3 个答案:

答案 0 :(得分:4)

您的代码效果很好:

   public static void main(String[] args) {

        int countZero = 0;
        int countOne = 0;
        int countTwo = 0;
        int countThree = 0;

        for(int i=0; i<400000; i++){            
            int randno =  (int)(Math.random() * ((3) + 1));

            if(randno == 0){
                countZero++;
            }
            else if(randno == 1){
                countOne++;
            }
            else if(randno == 2){
                countTwo++;
            }
            else if(randno == 3){
                countThree++;
            }           
        }

        System.out.println("Zero: " + countZero);
        System.out.println("One: " + countOne);
        System.out.println("Two: " + countTwo);
        System.out.println("Three: " + countThree);


    }

输出:

Zero: 99683
One: 99793
Two: 100386
Three: 100138

答案 1 :(得分:1)

为了进行比较,我将您的方法放在this之后。

public class Test {
    private static int min = 0;
    private static int max = 3;
    private static Map<Integer, Integer> count = new HashMap<>();

    public static void main(String[] args) {
        OptionOne();

        System.out.println("Option One: ");
        for (Entry<Integer, Integer> entry : count.entrySet()) {
            System.out.println(entry.getKey() + "\t " + entry.getValue());
        }

        count = new HashMap<Integer, Integer>();
        OptionTwo();
        System.out.println("\nOption Two:");
        for (Entry<Integer, Integer> entry : count.entrySet()) {
            System.out.println(entry.getKey() + "\t " + entry.getValue());
        }
    }

    private static void OptionOne() {
        for (int i = 0; i < 800000; i++) {
            int number = min + (int) (Math.random() * ((max - min) + 1));
            if (count.containsKey(number)) {
                int sofar = count.get(number) + 1;
                count.put(number, sofar);
            } else {
                count.put(number, 1);
            }
        }
    }

    private static void OptionTwo() {
        for (int i = 0; i < 800000; i++) {
            int number = (int) (Math.random() * 4);
            if (count.containsKey(number)) {
                int sofar = count.get(number) + 1;
                count.put(number, sofar);
            } else {
                count.put(number, 1);
            }
        }
    }

输出:

  

选项一:
  0 199853
  1 200118
  2 200136
  3 199893

     

选项二:
  0 199857
  1 200214
  2 199488
  3 200441

结论:您的方法有效。也许你的样本量不够?

答案 2 :(得分:1)

使用随机数字,您可以随机获得看似非随机的内容。无法保证所有序列都是随机的。 http://vanillajava.blogspot.com/2011/10/randomly-no-so-random.html

对于正确的随机种子,您可以看似获得非随机序列,但这些序列与其他序列一样可能。

Random random = new Random(441287210);
for (int i = 0; i < 10; i++)
    System.out.print(random.nextInt(10)+" ");
}

打印

1 1 1 1 1 1 1 1 1 1

Random random = new Random(-6732303926L);
for(int i = 0; i < 10; i++)
    System.out.println(random.nextInt(10)+" ");
}

打印

0 1 2 3 4 5 6 7 8 9

最后

public static void main(String ... args) {
    System.out.println(randomString(-229985452)+' '+randomString(-147909649));
}

public static String randomString(int seed) {
    Random rand = new Random(seed);
    StringBuilder sb = new StringBuilder();
    for (int i = 0; ; i++) {
        int n = rand.nextInt(27);
        if (n == 0) break;
        sb.append((char) ('`' + n));
    }
    return sb.toString();
}

打印

hello world