当前位置:首页 > 牛客 > 牛客4580题解:动态规划求解网格路径概率问题(C++代码实现)

牛客4580题解:动态规划求解网格路径概率问题(C++代码实现)

6小时前

牛客4580题解:动态规划求解网格路径概率问题(C++代码实现) 动态规划 概率计算 网格路径 牛客4580题 C++代码优化 第1张

一、题目解读

牛客4580题要求在一个n×m的网格中计算从起点(1,1)到终点(n,m)的概率。网格中存在障碍物(标记为坏点),路径只能向右或向下移动。到达终点时,若处于边界位置,概率转移规则不同:下边界时向右概率为0.5,右边界时向下概率为0.5,普通位置则各方向概率均为0.5。题目需要输出最终到达终点的概率,保留两位小数。

二、解题思路

该题目采用动态规划(DP)求解。核心思想是将问题分解为子问题,通过递推计算每个位置到达终点的概率。定义状态dp[i][j]为从起点到位置(i,j)的概率。根据路径规则,每个位置的概率由上方和左方位置的概率推导而来,需特殊处理边界和障碍物情况。通过遍历网格,逐步更新状态值,最终得到终点概率。

三、解题步骤

1. 输入处理:读取网格大小n、m及坏点数量k,标记坏点位置(bad数组)。

2. 初始化:起点概率dp[1][1]设为1,其他位置初始为0。

3. 状态转移:

○ 若当前位置为坏点,概率置0;

○ 若为终点,概率由上方和左方概率相加;

○ 若为下边界,上方概率1 + 左方概率0.5;

○ 若为右边界,上方概率 + 左方概率*0.5;

○ 普通位置则上方0.5 + 左方0.5。

4. 输出结果:固定小数点后两位输出dp[n][m]。

四、代码和注释

#include <iostream>
#include <vector>
#include <iomanip>
using namespace std;

int main() {
    int n, m, k;
    while (cin >> n >> m >> k) {  // 处理多组数据
        vector<vector<bool>> bad(n+1, vector<bool>(m+1, false));  // 标记坏点
        vector<vector<double>> dp(n+1, vector<double>(m+1, 0.0));  // 概率DP数组
        
        // 标记蘑菇位置
        while (k--) {
            int x, y;
            cin >> x >> y;
            bad[x][y] = true;  // 标记为坏点
        }
        
        dp[1][1] = 1.0;  // 起点概率为1
        
        for (int i = 1; i <= n; ++i) {  // 行遍历
            for (int j = 1; j <= m; ++j) {  // 列遍历
                if (i == 1 && j == 1) continue;  // 跳过起点
                if (bad[i][j]) {  // 坏点概率置0
                    dp[i][j] = 0.0;
                    continue;
                }
                
                // 处理边界情况
                if (i == n && j == m) {  // 终点
                    dp[i][j] = dp[i-1][j] + dp[i][j-1];
                } else if (i == n) {  // 下边界
                    dp[i][j] = dp[i-1][j] * 0.5 + dp[i][j-1];
                } else if (j == m) {  // 右边界
                    dp[i][j] = dp[i-1][j] + dp[i][j-1] * 0.5;
                } else {  // 普通位置
                    dp[i][j] = dp[i-1][j] * 0.5 + dp[i][j-1] * 0.5;
                }
            }
        }
        
        cout << fixed << setprecision(2) << dp[n][m] << endl;  // 输出终点概率
    }
    return 0;
}

五、总结

本题关键在于动态规划的状态定义与边界条件的处理。通过明确概率转移规则,结合网格遍历,高效计算出终点概率。代码中利用bad数组标记障碍物,避免了无效路径的计算,而边界条件的特殊处理保证了概率推导的正确性。该解法具有清晰逻辑和良好可扩展性,适用于类似网格路径概率问题。

原创内容 转载请注明出处

分享给朋友:

相关文章

力扣746:三步通关最小花费爬楼梯

力扣746:三步通关最小花费爬楼梯

题目解析:站在楼梯的某个台阶时,需要支付当前台阶对应的体力值cost[i],之后可以选择向上爬1或2个台阶。最终目标是到达‌楼层顶部‌(即数组末尾之后的位置),且初始位置可选择下标0或1的台阶作为起点...

IOI 1994 洛谷1216:如何用动态规划高效解决数字三角形问题?附完整代码解析

IOI 1994 洛谷1216:如何用动态规划高效解决数字三角形问题?附完整代码解析

题目重解给定一个由数字组成的三角形结构,从顶部出发,每次可以移动到下方相邻的数字,最终到达底部。我们的目标是找到一条路径,使得路径上经过的数字总和最大。这个问题在实际中有许多应用场景,如最优路径规划、...

GESP2023年六级真题解析:动态规划解决小杨买饮料问题(洛谷3873)

GESP2023年六级真题解析:动态规划解决小杨买饮料问题(洛谷3873)

一、题目解读小杨买饮料是GESP 2023年六级认证考试中的一道经典动态规划题目,考察学生对背包问题的理解和应用能力。题目描述小杨需要购买n种饮料,每种饮料有特定的体积w和价格v,他要在不超过容量l的...

2024年GESP五级武器强化(洛谷B4071)解题代码C++版

2024年GESP五级武器强化(洛谷B4071)解题代码C++版

一、题目解读    2024年GESP(青少年软件编程能力等级考试)五级中的“武器强化”(洛谷平台题目编号B4071)是一道典型的算法优化问题。题目要求通过合理...

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

CSP-J 2019纪念品题解(洛谷P5662):动态规划+完全背包问题的实战应用

一、题目解读2019年CSP-J的“纪念品”问题(对应洛谷P5662)要求玩家在T天内通过买卖纪念品最大化金币收益。每天可交易N种商品,需计算最优策略下的最终金币数。题目强调动态规划思维与资源分配优化...

NOIP 2008火柴棒等式题解(C++代码实现)  动态规划与枚举算法详解

NOIP 2008火柴棒等式题解(C++代码实现) 动态规划与枚举算法详解

一、题目解读火柴棒等式问题(NOIP 2008,洛谷P1149)要求使用给定数量的火柴棒,构造形如 A + B = C 的等式,其中A、B、C均为整数,且火柴棒总数恰好等于输入值。需统计符合条件的等式...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。