每日一题--n个骰子的点数

基于循环求骰子总数:用两个数组来存储骰子点数出现的每一个总和出现的次数:
第一个数组中的第n个数字代表骰子点数和为n出现的次数
第二个数组的第n个数字设为前一个数组对应的第n-1,第n-2,第n-3,n-4,n-5,n-6之和
解释:一次循环中第一个数组中的第n个数字代表骰子点数和为n出现的次数
下一次循环,我们加上一个新的骰子,此时和为n的骰子出现的次数应该为上一次循环中骰子点数和为n-1,n-1……n-6的次数总和(第二个数组的作用)
在一轮循环中,一个数组的第n项等于另一数组的前n-1,n-2,n-3,……n-6项,在下一轮循环中,交换这两个数组,通过改变flag来实现,再次循环
代码:

package six;

/**问题描述:n个骰子朝上一面和为s,输入n,打印出s的所有可能的值出现的概率
 * 统计出每一个点数出现的次数/6^n ,就能求出每个点数的概率
 * Created by lxq on 2017/9/17.
 */
public class Problem1 {
    public void printProbability(int number){
        if(number<1)
            return;
        int maxValue = 6;
        int Probabilities[][] = new int[2][];
        //第一个数组:第n个数字代表骰子和为n出现的次数
        //暂存
        Probabilities[0] = new int[maxValue*number+1];
        //第二个数组:加上一个骰子后
        // 第n个数字设为前一个数组对应的第n-1,第n-2,第n-3,n-4,n-5,n-6之和
        Probabilities[1] = new int[maxValue*number+1];
        int flag = 0;
        //一个骰子
        for(int i=1;i<=maxValue;i++){
            Probabilities[flag][i] = 1;
        }
        //每增加一个骰子,k代表骰子的数目
        for(int k=2;k<=number;k++){
            //骰子点数最小为k,所以数组前k项为0
            for(int i=0;i1-flag][i] = 0;
            }
            for(int i=k;i<=maxValue*k;i++){
                Probabilities[1-flag][i] = 0; //初始化
                //这里 j<=i&&j
                for(int j=1;j<=i&&j<=maxValue;j++) //……[0][n-6]
                    Probabilities[1-flag][i] += Probabilities[flag][i-j];
            }
            flag = 1-flag;
        }
        double total = Math.pow(maxValue,number);
        for(int i=number;i<=maxValue*number;i++){
            double ratio = (double) Probabilities[flag][i]/total;
            System.out.println(i+"");
            System.out.println(ratio);
        }

    }
}

你可能感兴趣的:(算法与数据结构)