当前位置: 首页 > news >正文

60. n 个骰子的点数【难】


comments: true
difficulty: 简单
edit_url: https://github.com/doocs/leetcode/edit/main/lcof/%E9%9D%A2%E8%AF%95%E9%A2%9860.%20n%E4%B8%AA%E9%AA%B0%E5%AD%90%E7%9A%84%E7%82%B9%E6%95%B0/README.md

面试题 60. n 个骰子的点数

题目描述

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

 

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

 

示例 1:

输入: 1
输出: [0.16667,0.16667,0.16667,0.16667,0.16667,0.16667]

示例 2:

输入: 2
输出: [0.02778,0.05556,0.08333,0.11111,0.13889,0.16667,0.13889,0.11111,0.08333,0.05556,0.02778]

 

限制:

1 <= n <= 11

解法

1.题目不太好理解,可以先看这个题:https://www.acwing.com/problem/content/76/
2.促进一点理解的题解:https://leetcode.cn/problems/nge-tou-zi-de-dian-shu-lcof/solutions/637778/jian-zhi-offer-60-n-ge-tou-zi-de-dian-sh-z36d/

方法一:动态规划

我们定义 f [ i ] [ j ] f[i][j] f[i][j] 表示投掷 i i i 个骰子,点数和为 j j j 的方案数。那么我们可以写出状态转移方程:

f [ i ] [ j ] = ∑ k = 1 6 f [ i − 1 ] [ j − k ] f[i][j] = \sum_{k=1}^6 f[i-1][j-k] f[i][j]=k=16f[i1][jk]

其中 k k k 表示当前骰子的点数,而 f [ i − 1 ] [ j − k ] f[i-1][j-k] f[i1][jk] 表示投掷 i − 1 i-1 i1 个骰子,点数和为 j − k j-k jk 的方案数。

初始条件为 f [ 1 ] [ j ] = 1 f[1][j] = 1 f[1][j]=1,表示投掷一个骰子,点数和为 j j j 的方案数为 1 1 1

最终,我们要求的答案即为 f [ n ] [ j ] 6 n \frac{f[n][j]}{6^n} 6nf[n][j],其中 n n n 为骰子个数,而 j j j 的取值范围为 [ n , 6 n ] [n, 6n] [n,6n]

时间复杂度 O ( n 2 ) O(n^2) O(n2),空间复杂度 O ( 6 n ) O(6n) O(6n)。其中 n n n 为骰子个数。

Python3
class Solution:def dicesProbability(self, n: int) -> List[float]:f = [[0] * (6 * n + 1) for _ in range(n + 1)] #最大情况:投掷n个骰子,最大和为6nfor j in range(1, 7):f[1][j] = 1 #投掷1个骰子,各个和的方案数都是1for i in range(2, n + 1):for j in range(i, 6 * i + 1): #1)投掷i个筛子,和的范围为[i,6*i]for k in range(1, 7): #2)核心递推:方案数的累加if j - k >= 0:f[i][j] += f[i - 1][j - k]m = pow(6, n) #3)n个骰子的总方案数return [f[n][j] / m for j in range(n, 6 * n + 1)]
Java
class Solution {public double[] dicesProbability(int n) {int[][] f = new int[n + 1][6 * n + 1];for (int j = 1; j <= 6; ++j) {f[1][j] = 1;}for (int i = 2; i <= n; ++i) {for (int j = i; j <= 6 * i; ++j) {for (int k = 1; k <= 6; ++k) {if (j >= k) {f[i][j] += f[i - 1][j - k];}}}}double m = Math.pow(6, n);double[] ans = new double[5 * n + 1];for (int j = n; j <= 6 * n; ++j) {ans[j - n] = f[n][j] / m;}return ans;}
}
C++
class Solution {
public:vector<double> dicesProbability(int n) {int f[n + 1][6 * n + 1];memset(f, 0, sizeof f);for (int j = 1; j <= 6; ++j) {f[1][j] = 1;}for (int i = 2; i <= n; ++i) {for (int j = i; j <= 6 * i; ++j) {for (int k = 1; k <= 6; ++k) {if (j >= k) {f[i][j] += f[i - 1][j - k];}}}}vector<double> ans;double m = pow(6, n);for (int j = n; j <= 6 * n; ++j) {ans.push_back(f[n][j] / m);}return ans;}
};
Go
func dicesProbability(n int) (ans []float64) {f := make([][]int, n+1)for i := range f {f[i] = make([]int, 6*n+1)}for j := 1; j <= 6; j++ {f[1][j] = 1}for i := 2; i <= n; i++ {for j := i; j <= 6*i; j++ {for k := 1; k <= 6; k++ {if j >= k {f[i][j] += f[i-1][j-k]}}}}m := math.Pow(6, float64(n))for j := n; j <= 6*n; j++ {ans = append(ans, float64(f[n][j])/m)}return
}
JavaScript
/*** @param {number} n* @return {number[]}*/
var dicesProbability = function (n) {const f = Array.from({ length: n + 1 }, () => Array(6 * n + 1).fill(0));for (let j = 1; j <= 6; ++j) {f[1][j] = 1;}for (let i = 2; i <= n; ++i) {for (let j = i; j <= 6 * i; ++j) {for (let k = 1; k <= 6; ++k) {if (j >= k) {f[i][j] += f[i - 1][j - k];}}}}const ans = [];const m = Math.pow(6, n);for (let j = n; j <= 6 * n; ++j) {ans.push(f[n][j] / m);}return ans;
};
C#
public class Solution {public double[] DicesProbability(int n) {int[,] f = new int[n + 1, 6 * n + 1];for (int j = 1; j <= 6; ++j) {f[1, j] = 1;}for (int i = 2; i <= n; ++i) {for (int j = i; j <= 6 * i; ++j) {for (int k = 1; k <= 6; ++k) {if (j >= k) {f[i, j] += f[i - 1, j - k];}}}}double m = Math.Pow(6, n);double[] ans = new double[5 * n + 1];for (int j = n; j <= 6 * n; ++j) {ans[j - n] = f[n, j] / m;}return ans;}
}
Swift
class Solution {func dicesProbability(_ n: Int) -> [Double] {var f = Array(repeating: Array(repeating: 0, count: 6 * n + 1), count: n + 1)for j in 1...6 {f[1][j] = 1}if n > 1 {for i in 2...n {for j in i...(6 * i) {for k in 1...6 {if j >= k {f[i][j] += f[i - 1][j - k]}}}}}var m = 1.0for _ in 0..<n {m *= 6.0}var ans = Array(repeating: 0.0, count: 5 * n + 1)for j in n...(6 * n) {ans[j - n] = Double(f[n][j]) / m}return ans}
}

方法二:动态规划(空间优化)

我们可以发现,上述方法中的 f [ i ] [ j ] f[i][j] f[i][j] 的值仅与 f [ i − 1 ] [ j − k ] f[i-1][j-k] f[i1][jk] 有关,因此我们可以使用滚动数组的方式,将空间复杂度优化至 O ( 6 n ) O(6n) O(6n)

Python3
class Solution:def dicesProbability(self, n: int) -> List[float]:f = [0] + [1] * 6for i in range(2, n + 1):g = [0] * (6 * i + 1)for j in range(i, 6 * i + 1):for k in range(1, 7):if 0 <= j - k < len(f):g[j] += f[j - k]  #仅仅与投掷n-1个骰子的方案数 数组有关f = gm = pow(6, n)return [f[j] / m for j in range(n, 6 * n + 1)]
Java
class Solution {public double[] dicesProbability(int n) {int[] f = new int[7];Arrays.fill(f, 1);f[0] = 0;for (int i = 2; i <= n; ++i) {int[] g = new int[6 * i + 1];for (int j = i; j <= 6 * i; ++j) {for (int k = 1; k <= 6; ++k) {if (j - k >= 0 && j - k < f.length) {g[j] += f[j - k];}}}f = g;}double m = Math.pow(6, n);double[] ans = new double[5 * n + 1];for (int j = n; j <= 6 * n; ++j) {ans[j - n] = f[j] / m;}return ans;}
}
Go
func dicesProbability(n int) (ans []float64) {f := make([]int, 7)for i := 1; i <= 6; i++ {f[i] = 1}for i := 2; i <= n; i++ {g := make([]int, 6*i+1)for j := i; j <= 6*i; j++ {for k := 1; k <= 6; k++ {if j-k >= 0 && j-k < len(f) {g[j] += f[j-k]}}}f = g}m := math.Pow(6, float64(n))for j := n; j <= 6*n; j++ {ans = append(ans, float64(f[j])/m)}return
}
JavaScript
/*** @param {number} num* @return {number[]}*/
var dicesProbability = function (n) {let f = Array(7).fill(1);f[0] = 0;for (let i = 2; i <= n; ++i) {let g = Array(6 * i + 1).fill(0);for (let j = i; j <= 6 * i; ++j) {for (let k = 1; k <= 6; ++k) {if (j - k >= 0 && j - k < f.length) {g[j] += f[j - k];}}}f = g;}const ans = [];const m = Math.pow(6, n);for (let j = n; j <= 6 * n; ++j) {ans.push(f[j] / m);}return ans;
};
C#
public class Solution {public double[] DicesProbability(int n) {int[] f = new int[7];for (int i = 1; i <= 6; ++i) {f[i] = 1;}f[0] = 0;for (int i = 2; i <= n; ++i) {int[] g = new int[6 * i + 1];for (int j = i; j <= 6 * i; ++j) {for (int k = 1; k <= 6; ++k) {if (j - k >= 0 && j - k < f.Length) {g[j] += f[j - k];}}}f = g;}double m = Math.Pow(6, n);double[] ans = new double[5 * n + 1];for (int j = n; j <= 6 * n; ++j) {ans[j - n] = f[j] / m;}return ans;}
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 裸金属服务器怎么实现算力共享,裸金属服务器提供者怎么做,租户怎样使用,共享平台需要搭建什么
  • 深入理解Appium定位策略与元素交互
  • FISC安全対策基準
  • 网络工程师考试真题及解析
  • 代码随想录算法训练营第28天 | 第九章动态规划 part01
  • 深度解读:从新手到专业,大模型开发者知识技能养成之路
  • 大数据开发:可视化组件Redash安装部署
  • Linux df命令详解,Linux查看磁盘使用情况
  • 力扣: 翻转字符串里的单词
  • Django创建模型
  • 从零开始学数据结构系列之第六章《排序简介》
  • 三维坐标变换
  • linux c++ 通信架构 笔记(14) 第三章 Nginx 开发初步:守护进程的信号使用,介绍 nginx 的选项与信号,后台进程与守护进程的区别
  • websocket 和sip 在协议层面有哪些区别,为什么要各自这样设置协议
  • redis的事务与管道有什么不同?
  • [分享]iOS开发 - 实现UITableView Plain SectionView和table不停留一起滑动
  • CSS3 变换
  • gcc介绍及安装
  • iOS高仿微信项目、阴影圆角渐变色效果、卡片动画、波浪动画、路由框架等源码...
  • JavaScript HTML DOM
  • LeetCode18.四数之和 JavaScript
  • leetcode386. Lexicographical Numbers
  • mongo索引构建
  • mysql 数据库四种事务隔离级别
  • Python_OOP
  • Quartz初级教程
  • use Google search engine
  • yii2中session跨域名的问题
  • 前端之React实战:创建跨平台的项目架构
  • #Datawhale AI夏令营第4期#AIGC文生图方向复盘
  • #gStore-weekly | gStore最新版本1.0之三角形计数函数的使用
  • #NOIP 2014# day.1 T3 飞扬的小鸟 bird
  • #QT 笔记一
  • (0)Nginx 功能特性
  • (1)SpringCloud 整合Python
  • (3)选择元素——(14)接触DOM元素(Accessing DOM elements)
  • (C#)if (this == null)?你在逗我,this 怎么可能为 null!用 IL 编译和反编译看穿一切
  • (k8s)Kubernetes 从0到1容器编排之旅
  • (附源码)计算机毕业设计SSM智慧停车系统
  • (南京观海微电子)——示波器使用介绍
  • (牛客腾讯思维编程题)编码编码分组打印下标题目分析
  • (十)【Jmeter】线程(Threads(Users))之jp@gc - Stepping Thread Group (deprecated)
  • (四)搭建容器云管理平台笔记—安装ETCD(不使用证书)
  • (一)认识微服务
  • (转)ABI是什么
  • (转)四层和七层负载均衡的区别
  • .[hudsonL@cock.li].mkp勒索加密数据库完美恢复---惜分飞
  • .form文件_SSM框架文件上传篇
  • .NET CF命令行调试器MDbg入门(一)
  • .net core控制台应用程序初识
  • .net 反编译_.net反编译的相关问题
  • .NET 某和OA办公系统全局绕过漏洞分析
  • .NET/C# 判断某个类是否是泛型类型或泛型接口的子类型
  • .NET开源项目介绍及资源推荐:数据持久层
  • //usr/lib/libgdal.so.20:对‘sqlite3_column_table_name’未定义的引用