剑指 Offer !!! 60. n个骰子的点数

剑指 Offer 60. n个骰子的点数

把n个骰子扔在地上,所有骰子朝上一面的点数之和为s。输入n,打印出s的所有可能的值出现的概率。

你需要用一个浮点数数组返回答案,其中第 i 个元素代表这 n 个骰子所能掷出的点数集合中第 i 小的那个的概率。

示例 1:

输入: 1

输出: 0.16667,0.16667,0.16667,0.16667,0.16667,0.16667

思路:

使用动态规划

java 复制代码
 public double[] dicesProbability(int n) {
         int[][] dp = new int[n+1][6*n+1];// when we have n dices, the maxSum we could get is 6*n
 // dp[i][j] means #methods of using i dices to obtain sum j

         for(int j=1;j<=6;j++){
             dp[1][j]=1;
         }

         for(int i=2;i<=n;i++){
             // now, we have i dices available
             for(int j=i;j<=6*i;j++){
                 // when sum is j...
                 // the current dice could take value of 1,2,...,6
                 for(int k=1;k<=6 && k<j;k++){
                     dp[i][j]+=dp[i-1][j-k];
                 }
             }
         }

         double[] ans = new double[5*n+1];// compute the final solution
         for(int j=n;j<=6*n;j++){
             ans[j-n]=(double)dp[n][j]/Math.pow(6,n);
         }
         return ans;
     }

考虑到上述二维的动态规划表dpij仅依赖于上一行dpi-1..., 可以做压缩, 也就是把使用两个数组交替着完成二维表dp的更新,直到更新出dp表的最后一行.

下面的代码参考了力扣K神的代码,只是为了便于理解,我做了小小的改动,也就是
dp[j]表示前i-1个 骰子和为j时的概率.
tmp[j]表示前i个 骰子和为j时的概率;

java 复制代码
public double[] dicesProbability(int n) {
		// 前一个骰子和的概率分布
        double[] dp = new double[6+1];
        Arrays.fill(dp,1.0/6.0);
        dp[0]=0;
        
        for(int i=2;i<=n;i++){//轮到第i个骰子参与加和
            double[] tmp = new double[6*i+1];// 前i个骰子和的分布概率
            for(int j=1;j<dp.length;j++){
                for(int k=1;k<=6;k++){// 第i个骰子取值为k的概率为1/6
                    tmp[j+k]+=dp[j]/6.0;
                }
            }
            dp=tmp;
        }
		
		//从dp中整理结果
		// 此时dp的前一段是无用的,如当n=3时dp[0..2]是无用的,因为三个骰子的最小和为3.
        double[] res = new double[5*n+1];
        int index=0;
        for(double i:dp){
            if(i!=0)break;
            index++;
        }// 找到dp中第一个不为零的值的位置index,然后开始誊写到res
        int i=0;
        for(;index<dp.length;index++){
            res[i++]=dp[index];
        }
        return res;

    }
相关推荐
KaMeidebaby2 小时前
卡梅德生物技术快报|PD1 单克隆抗体定制配套 N 糖全谱质控开发
前端·人工智能·算法·数据挖掘·数据分析
8Qi83 小时前
LeetCode 235. 二叉搜索树的最近公共祖先(LCA)
算法·leetcode·二叉树·递归·二叉搜索树·lca·迭代
bIo7lyA8v3 小时前
算法稳定性分析中的随机扰动建模的技术8
算法
科研online4 小时前
基于多源数据和XGBoost-SHAP分析中国大陆绿地碳汇空间变异影响因素的非线性相关性与尺度差异
算法·学习方法
Cthy_hy4 小时前
拓扑排序超详解:原理 + Kahn 贪心算法
python·算法·贪心算法
三品吉他手会点灯5 小时前
C语言学习笔记 - 43.运算符与表达式 - 运算符1 - 运算符的分类和简单介绍
c语言·笔记·学习·算法
VkN2X2X4b5 小时前
算法复杂度的实验验证与误差分析的技术8
算法
其利天下技术5 小时前
风扇灯无刷电机自适应算法实战指南
算法·cocos2d·无刷电机自适应算法·bldc驱动自适应算法·其利无刷电机驱动算法
8Qi85 小时前
LeetCode 494:目标和(Target Sum)—— 题解 ✅
算法·leetcode·职场和发展·动态规划·01背包
hujinyuan201606 小时前
2026年3月 中国电子学会青少年软件编程(Python)三级考试试卷 真题及答案
java·python·算法