13

我需要在 Java 中随机生成一个包含 7 个插槽的数组。所有这些插槽的值必须至少为 1,但结合起来,总值必须是另一个定义的数字。它们也都需要是一个 int 值,没有 1.5 或 0.9816465684646 数字。例子:

int a=10;

int[] ar = new int[7]
ar[0] = 1
ar[1] = 1
ar[2] = 2
ar[3] = 2
ar[4] = 1
ar[5] = 2
ar[6] = 1

我希望它生成类似的东西,但如果 int a=15,所有数字按任意顺序总计为 15

4

4 回答 4

40

生成添加到给定总和的 N 个随机数的标准方法是将总和视为数轴,在该线上生成 N-1 个随机点,对它们进行排序,然后使用点之间的差异作为最终值。要获得最小值 1,首先从总和中减去 N,运行给定的算法,然后将 1 加回每个段。

public class Rand {
    public static void main(String[] args) {
        int count = 8;
        int sum = 100;
        java.util.Random g = new java.util.Random();

        int vals[] = new int[count];
        sum -= count;

        for (int i = 0; i < count-1; ++i) {
            vals[i] = g.nextInt(sum);
        }
        vals[count-1] = sum;

        java.util.Arrays.sort(vals);
        for (int i = count-1; i > 0; --i) {
            vals[i] -= vals[i-1];
        }
        for (int i = 0; i < count; ++i) { ++vals[i]; }

        for (int i = 0; i < count; ++i) {
            System.out.printf("%4d", vals[i]);
        }
        System.out.printf("\n");
    }
}
于 2013-09-03T20:15:29.013 回答
14

实现一致性的一个好方法是,例如,将a = 15单元填充到一个 8 元素数组中:

  1. 将 1 放入数组中的每个元素中,因为这是您的要求,您现在还有 7 个值可供分配
  2. 在 0 和数组的最大索引之间滚动一个随机数,然后将 1 添加到该元素,然后从 7 中减去 1。这样做直到 7 下降到零。

这样,您将通过让每个元素的最小值为 1 来满足您的最低条件。然后您以完全随机的方式分配剩余的总数。

于 2013-09-03T19:50:43.480 回答
2

除了@Kon 所说的,您可以使用两个随机数而不是一个来获得更多随机性。那是:

Fill every element in the array with the value 1
valuesToDistribute = a - array.length-1
randomIndex = Roll a number between 0 and array.length-1
randomValue = Roll a number between 1 and valuesToDistribute
Add to randomIndex the value randomValue
Subtract randomValue from valuesToDistribute
Repeat until valuesToDistribute = 0
于 2013-09-03T20:18:48.060 回答
1

我的 java 很糟糕,所以我没有在这里提供实际代码,因为它可能是错误的。不过,我以前在 SQL 中做过这个确切的事情,所以我知道它有效......

  1. 让 Y 是您希望元素相加的总值
  2. 开始一个循环,变量 Z 从 1 到 X,其中 X 是数组中的数字元素(这里称为 AR)
  3. 在循环中,将 AR(Z) 设置为 1 和 Y-X+Z 之间的随机数
  4. 从 Y 中减去新值,所以 Y = Y - AR(Z)
  5. 结束循环:返回第 2 步,将 Z 前进 1
于 2013-09-03T20:18:11.697 回答