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

一、思路:题目要求是计算掷n个骰子时,所有可能点数出现的概率。这道题可以用动态规划dp来做。

1.投掷n个骰子时,可能出现的总和最小值是n * 1,可能出现的总和最大值是n * 6。

2.从最小值到最大值中,所有连续的整数都是可以取到的,这是因为可以通过每次增加1点的点数的方式来调整骰子点数,从而得到中间的所有整数。因此所有的总和值构成了一个连续的整数区间。

3.因此所有可能出现的点数的总数量 = 6n - n + 1 = 5n + 1。

二、动态规划dp

1.确定dp数组及其下标的含义:dpij表示投掷i个骰子,点数总和恰好等于j的概率。

2.确定递推公式:

可知i个骰子的情况可以由i - 1个骰子的情况推出。

(1)因此dpij = dpi - 1j - 6 + dpi - 1\[j - 5 + dpi - 1j - 4 + dpi - 1j - 3 + dpi - 1j - 2 + dpi - 1j - 1

(2)即:dpij = dpi - 1j - k for k = 1 to 6 。

3.dp数组如何初始化:dp1j=1/6 for j = 1 to 6。表示使用1个骰子的时候,点数总和恰好等于1,2,3,4,5,6的概率均为1/6。

4.确定遍历顺序:从前往后遍历即可。在二维dp中遍历顺序其实不太敏感。因为我们在用二维数组存储所有状态,每个状态只依赖上一行的数据。

附代码:

java 复制代码
class Solution {
    public double[] dicesProbability(int n) {
        // dp[i][j] 表示投掷 i 个骰子,点数和为 s 的概率
        // i 的范围:0 ~ n
        // j 的范围:0 ~ 6*n(但实际有效范围是 i ~ 6*i)
        double[][] dp = new double[n + 1][6 * n + 1];

        // 初始化:1个骰子的情况
        for (int j = 1; j <= 6; j++) {
            dp[1][j] = 1.0 / 6.0;
        }

        // 从第2个骰子开始递推
        for (int i = 2; i <= n; i++) {
            // 当前i个骰子的点数和范围:i ~ 6*i
            for (int j = i; j <= 6 * i; j++) {
                // 第i个骰子掷出k(1~6),则前i-1个骰子的和为 j-k
                for (int k = 1; k <= 6; k++) {
                    int prevSum = j - k;
                    // 检查前i-1个骰子的和是否有效,即是否大于i - 1(对应点数和全为1的情况)
                    if (prevSum >= i - 1){
                        // 前i - 1个骰子点数总和等于j - k的概率 * 第i个骰子点数为1/6的概率
                        // 因为k会从1-6中取值,所以要把这6种可能累加
                        dp[i][j] += dp[i - 1][prevSum] * (1.0 / 6.0);
                    }
                }
            }
        }

        // 提取结果:n个骰子的有效范围是 n ~ 6n
        int total = 5 * n + 1;
        double[] result = new double[total];
        for (int i = 0; i < total; i++) {
            // 索引转换,n的骰子时的点数总和范围为n到6n,对应j
            // 对应的dp数组的位置即为dp[n][n]到dp[n][6n]
            result[i] = dp[n][n + i];
        }
        return result;
    }
}

ACM模式:

java 复制代码
import java.util.Scanner;

class Solution {
    public double[] dicesProbability(int n) {
        // dp[i][j] 表示投掷 i 个骰子,点数和为 s 的概率
        // i 的范围:0 ~ n
        // j 的范围:0 ~ 6*n(但实际有效范围是 i ~ 6*i)
        double[][] dp = new double[n + 1][6 * n + 1];

        // 初始化:1个骰子的情况
        for (int j = 1; j <= 6; j++) {
            dp[1][j] = 1.0 / 6.0;
        }

        // 从第2个骰子开始递推
        for (int i = 2; i <= n; i++) {
            // 当前i个骰子的点数和范围:i ~ 6*i
            for (int j = i; j <= 6 * i; j++) {
                // 第i个骰子掷出k(1~6),则前i-1个骰子的和为 j-k
                for (int k = 1; k <= 6; k++) {
                    int prevSum = j - k;
                    // 检查前i-1个骰子的和是否有效,即是否大于i - 1(对应点数和全为1的情况)
                    if (prevSum >= i - 1){
                        // 前i - 1个骰子点数总和等于j - k的概率 * 第i个骰子点数为1/6的概率
                        // 因为k会从1-6中取值,所以要把这6种可能累加
                        dp[i][j] += dp[i - 1][prevSum] * (1.0 / 6.0);
                    }
                }
            }
        }

        // 提取结果:n个骰子的有效范围是 n ~ 6n
        int total = 5 * n + 1;
        double[] result = new double[total];
        for (int i = 0; i < total; i++) {
            // 索引转换,n的骰子时的点数总和范围为n到6n,对应j
            // 对应的dp数组的位置即为dp[n][n]到dp[n][6n]
            result[i] = dp[n][n + i];
        }
        return result;
    }
}

public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        int n = scanner.nextInt();
        scanner.close();

        Solution solution = new Solution();
        double[] result = solution.dicesProbability(n);

        for (int i = 0; i < result.length; i++) {
            System.out.printf("%.5f", result[i]);
            if (i < result.length - 1) {
                System.out.print(" ");
            }
        }
        System.out.println();
    }
}
相关推荐
QiLinkOS4 小时前
第三视觉理解徐玉生与他的商业活动(30)
大数据·c++·人工智能·算法·开源协议
疯狂打码的少年4 小时前
【操作系统】页面置换算法(OPT/FIFO/LRU)
算法
小O的算法实验室5 小时前
2026年CIE,优化客货协同运输:综合地铁系统的列车容量动态分配
算法
Coder_Shenshen6 小时前
西门子S7CommPlus协议鉴权算法原理与流程详解
网络·后端·算法
硕风和炜6 小时前
【LeetCode: 2492. 两个城市间路径的最小分数 + DFS】
java·算法·leetcode·深度优先·dfs·bfs·并查集
我是一颗柠檬7 小时前
【Java项目技术亮点】加权轮询负载均衡算法
java·算法·负载均衡
灯厂码农7 小时前
C语言动态内存分配完全指南(malloc、calloc、realloc、free)
java·c语言·算法
凯瑟琳.奥古斯特9 小时前
K次取反最大化数组和解法(力扣1005)
开发语言·c++·算法·leetcode·职场和发展
Jerry9 小时前
LeetCode 203. 移除链表元素
算法
地平线开发者9 小时前
征程 6 | 工具链 QAT ObserverBase 源码解析
算法