动态规划算法:10.路径问题_地下城游戏_C++

目录

[题目链接:174. 地下城游戏 - 力扣(LeetCode)](#题目链接:174. 地下城游戏 - 力扣(LeetCode))

一、题目解析

题目:​编辑

解析:

二、算法原理

1、状态表示

2、状态转移方程

状态转移方程推理:

3、初始化

dp表初始化:

特殊位置初始化:

4、填表顺序

5、返回值

三、编写代码


题目链接:174. 地下城游戏 - 力扣(LeetCode)

一、题目解析

题目:

解析:

我们由题目可以知道,我们骑士的最低血量就是我们要求得最小初始值,我们要求骑士拯救公主后血量在0以上,那么最小的时候是什么情况:

  • 如果拯救公主时,有守卫守护,那么我们希望拯救完公主后,血量为1
  • 如果拯救公主时,没有首位守护,或者有健康点数可以增加,那么我们希望拯救公主前,血量为1

并且在以上过程中,骑士的血量不可以降至0或者0以下

二、算法原理

1、状态表示

我们在状态标识的时候,一般都会创建一个数组dp,也就是我们所说的dp表,我们要做的就是把每一个状态的值填入这个表内,最终这个表内的某一个值可能就是我们要返回的值。

状态简单理解就是dp表内某一个值代表的含义。

如何确定状态表示

  • 题目要求

简单的题目里一般会给出

  • 经验+题目要求

越学越深入,动态规划也是熟能生巧,在题目中没有明显给出的时候,我们就要凭借自己做题的经验来确定,所以就需要我们大量的做题。

  • 分析问题的过程中,发现重复子问题

分析问题的过程中把重复子问题抽象成我们的状态表示,这个更难理解,一切的基础都是我们先对动态规划算法熟练运用。我也不懂,我们慢慢来。

综上:我们通常会以一个位置为结尾或者开始求得我们想要的答案

那我们的这道题得状态表示是什么样的:

根据经验,我们在此以一个位置为开始解题

为什么这次我们以一个位置为开始,但是之前都是以一个位置为结尾呢?

如果以一个位置为结尾解题,那么该点应该表示,到达该位置时需要的最小生命值 ,那么如果下一个位置的值是一个负数呢?我们只能保证通过该位置,但是并不能保证安全通过下一个位置, 所以我们以一个位置为开始解题,也就是从该位置开始,到达终点所需要的最小生命值

状态表示:dpij表示,从(i,j)位置开始,到达终点所需要的最小生命值

2、状态转移方程

确定状态表示之后我们就可以根据状态标识推出状态转移方程

状态转移方程是什么?

不讲什么复杂的,简单来说状态转移方程就是 dpij等于什么 dpij=?

这个就是状态转移方程,我们要做的,就是推出dpij等于什么

我们根据状态表示再结合题目+经验去推理转移方程,这一步也是我们整个解题过程中最难的一步

我们在这道题先简单了解下什么是状态转移方程,之后比较难的题目再细推

状态转移方程推理:

我们考虑从右下往左上 进行动态规划。令 dpij 表示从坐标 (i,j) 到终点所需的最小初始值 。换句话说,当我们到达坐标 (i,j) 时,如果此时我们的路径和不小于 dpij,我们就能到达终点

这样一来,我们就无需担心路径和的问题,只需要关注最小初始值。对于 dpij,求从位置(i,j)到达终点的最小值,我们只要关心 dpij+1 和 dpi+1j 的最小值 min,减去(i,j)位置的值,就是我们所求的dpij ,同时,初始值还必须大于等于 1。这样我们就可以得到状态转移方程:

状态转移方程:dpij=max(min(dpi+1j,dpij+1)−(i,j)位置的值,1)

3、初始化

我们创建dp表就是为了把他填满,我们初始化是为了防止在填表的过程中越界

怎么谈越界?

dp表初始化:

如果我们的dp表创建成与原表大小相同 ,那么在处理最后一个值时,其根本没有dpij+1与dpi+1j,这就会造成越界, 不仅如此,我们dp表的最后一行,也不会有dpi+1j,dp表的最后一列,也不会有dpij+1

所以我们就需要额外的增加一行一列,来保证我们填表的时候不越界

我们在对整个dp表初始化时,我们不能让新增的值影响原有数据填表,根据状态方程我们知道,新增的值会与原数据进行一个比较,从而选择较小的那个,所以我们在整体初始化时,把所有的值初始化为INT_MAX,这样就不会影响了。

特殊位置初始化:

我们填表的第一个值是原数据的最后一个值,也就是公主所在的位置,此时,其dpij+1与dpi+1j都是INT_MAX,并没有原数据在其中比较,所以根据状态转移方程我们知道,如果都是INT_MAX的话,该位置最终也会是INT_MAX,最终影响填表。该位置就是终点,这个位置的如果是正数的话,其dp值应该为1,如果是负数,那么它的dp值应该是(1-该位置的值)。所以为了保证该位置填表正确,不影响之后填表,我们需要把其dpij+1和dpi+1j设置为1即可。

4、填表顺序

从下到上,从右到左依次填表

5、返回值

dp表的第一个值,也就是dp00

三、编写代码

cpp 复制代码
class Solution {
public:
    int calculateMinimumHP(vector<vector<int>>& d) {
  int m =d.size(),n=d[0].size();
//1、创建dp表
  vector<vector<int>> dp(m+1,vector<int>(n+1,INT_MAX));
//2、特殊位置初始化
  dp[m-1][n]=1;
  dp[m][n-1]=1;
//3、填表
  for(int i=m-1;i>=0;i--)
  {
    for(int j=n-1;j>=0;j--)
    {
//这里我将状态转移方程拆开来写了
        dp[i][j]=min(dp[i][j+1],dp[i+1][j])-d[i][j];
        dp[i][j]=max(dp[i][j],1);
    }
  }
//4、返回值
  return dp[0][0];

    }
};
相关推荐
丘山望岳1 天前
藤萝垂序——二叉搜索树
开发语言·数据结构·c++
Qhappy1 天前
某里v2反混淆 codec 化路上踩到的两个隐蔽坑:被清零的 salt 与 opaque loop bound
javascript·算法
Hello world.Joey1 天前
吴恩达深度学习基础
人工智能·深度学习·神经网络·opencv·算法·机器学习·计算机视觉
水木流年追梦1 天前
大模型入门-大模型优化方法1
人工智能·学习·算法·机器学习·正则表达式
光电笑映1 天前
深入理解 ELF:从目标文件到程序加载的全过程
linux·运维·服务器·c++
lynnlovemin1 天前
【信息学竞赛专题】滑动窗口(尺取法)超全详解|C++模板+经典例题+避坑指南
开发语言·c++·算法·滑动窗口·信息学竞赛
不会C语言的男孩1 天前
VS Code 中搭建 C/C++ 开发环境(MSYS2 编译器)
c语言·c++
似水এ᭄往昔1 天前
【Qt】--Qt概述
开发语言·c++·qt
澈2071 天前
动态规划入门:从斐波那契到爬楼梯
c++·算法
月光船幽幽1 天前
四层门禁+自愈机制
人工智能·科技·动态规划·拓扑学