[dp4_路径问题] 下降路径最小和 | 最小路径和 | 地下城游戏
目录
1.下降路径最小和
题解
2.最小路径和
题解
3.地下城游戏
题解
做算法题的时候,谨记图画得越详细越好,思路想的越清晰越好,然后再用代码实现一下就好啦
1.下降路径最小和
链接:931. 下降路径最小和
给你一个 n x n 的 方形 整数数组 matrix ,请你找出并返回通过 matrix 的下降路径 的 最小和 。
下降路径 可以从第一行中的任何元素开始,并从每一行中选择一个元素。在下一行选择的元素和当前行所选元素最多相隔一列(即位于正下方或者沿对角线向左或者向右的第一个元素)。具体来说,位置 (row, col) 的下一个元素应当是 (row + 1, col - 1)、(row + 1, col) 或者 (row + 1, col + 1) 。

题解
- 给一个nxn矩阵,让返回从第一行下降到最后一行的最小和
- 如果当前位置是(row, col) 的下一个元素应当是
(row + 1, col - 1)、(row + 1, col) 或者 (row + 1, col + 1) 中选一个。
1.状态表示
- 经验 + 题目要求
- dp[i][j] 表示:到达 [i][j] 位置时,最小的下降路径
2.状态转移方程
- 根据最近一步,划分问题
- 到达[i][j] 位置有三种情况,从[i-1][j-1],[i-1][j],[i-1][j+1]任意一个位置都可以到。
- 但是我们要找的是到达 [i][j] 最小路径,那 [i-1][j-1],[i-1][j],[i-1][j+1]一定要是最小。
而dp[i][j] 表示:到达 [i][j] 位置时,最小的下降路径。然后在加上[i][j]本身的值。所以状态转移方程为
- dp[i][j]=min(dp[i-1][j-1],dp[i-1][j],dp[i-1][j+1])+m[i][j]
- 保证最小,就保证之前的每一步都是最小的,然后再加上这一步就好啦
- 就像我们做事一样,做好当下就好了,这样之后也一定会是好的~
3.初始化
如果我们开辟和原数组一样大小的dp表,第一行和第一列和最后一列填表的时候都会越界。
因此dp表可以多开一行多开两列。

- 虚拟节点里面的值,要保证后面填表的结果是正确的
- 下标的映射
虚拟节点的值应该放什么呢。
- 先不考虑多加的行和列。
- 没有这些的话,考虑这些填表越界的格子应该填多少。
- 先看第一行,从第一行开始到达第一行当前位置最小下降路径和,此时是没有选择的,只能是自己本身的值。
正好代入状态转移方程是没问题的,因此第一行虚拟节点的值应该填0。
- 再看第一列和最后一列,如果第一列和最后一轮还是0的话,可不可以?
- 如果第一列和最后一列是0的话,我们要的是三者中的最小值
- 在填表的时候就影响到上面经过计算得到的正确结果,因此第一行和第一列不能是0,我们可以给一个+∞。
- 因为dp表多加一行两列整是往右下移动一位的。
填表的时候要从dp表找到原始矩阵的值,必须i-1,j-1。
4.填表顺序
- 从上到下填写每一行,每一行从左往右
5.返回值
- 找到最后一行的最小值
class Solution {
public:int minFallingPathSum(vector<vector<int>>& matrix) {int m=matrix.size(),n=matrix[0].size();//多两列哦vector<vector<int>> dp(m+1,vector<int>(n+2,0));//因为每行是要选择最小值,所以两列要初始化为无穷大for(int i=1;i<=m;i++){dp[i][0]=dp[i][n+1]=INT_MAX;}//填表for(int i=1;i<=m;i++){for(int j=1;j<=n;j++){
dp[i][j]=min(min(dp[i-1][j],dp[i-1][j-1]),dp[i-1][j+1])
+matrix[i-1][j-1]; //不可以三个min//注意 下表映射//遍历 处理 上一行的-->推出这一行的}}int ret=INT_MAX;for(int j=1;j<=n;j++){ret=min(ret,dp[m][j]);}return ret;}
};
2.最小路径和
链接:64. 最小路径和
给定一个包含非负整数的 m x n 网格 grid ,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。
说明:每次只能向下或者向右移动一步。

题解
- 求左上角到右上角最小路径和。
- 注意只能向下向右移动!
1.状态表示
- 经验 + 题目要求
- dp[i][j] 表示:到达 [i][j] 位置时,最小路径和
2.状态转移方程
- 根据最近一步,划分问题
到达[i][j]有两种情况,[i-1][j],[i][j-1]都可以到,但是我们要找的是到达 [i][j] 最小路径
- 那[i-1][j],[i][j-1]一定要是最小。
- 而dp[i][j] 表示:到达 [i][j] 位置时,最小的下降路径。
然后在加上[i][j]本身的值。所以状态转移方程为
- dp[i][j]=min(dp[i-1][j],dp[i][j-1])+g[i][j]
3.初始化
- 填表时保证不越界
如果我们开辟和原数组一样大小的dp表,第一行和第一列填表的时候都会越界。
因此dp表可以多开一行一列。

- 虚拟节点里面的值,要保证后面填表的结果是正确的
- 下标的映射
虚拟节点的值应该放什么呢。
- 先不考虑多加的行和列。没有这些的话,考虑这些填表越界的格子应该填多少。
- 先看第一个格子,第一个格子的意思是dp[0][0]表示达到[0][0]位置的最小值,那没得选就是它本身也就是之前矩阵g[0][0]的值
- 这个格子的值是上一个格子和左边格子的最小值在加上自己本身
因此第一个格子上面和左边虚拟格子我们给0
剩下的虚拟格子也给0吗?
- 不行的。如果是0,那这个虚拟格子里的值就会参与取最小值的运算。
- 会导致填表错误。因此把剩下格子设置为+∞,只把[0][1]和[1][0]位置设置为0就好了。
当前你也可以直接把第一行和第一列初始化,但是用的更多的还是多开一行一列。
如果做的多,你会发现求最小值,就是dp表先给+∞,仅修改几个位置的值就可以了。
同样求最大值,dp表先给-∞,然后修改几个位置的值就可以了。
4.填表顺序
- 从上到下填写每一行,每一行从左往右
5.返回值
- 多开一行一列右下角就变成[m][n],所以返回dp[m][n]
class Solution {
public:int minPathSum(vector<vector<int>>& grid) {int m=grid.size(),n=grid[0].size();
//minvector<vector<int>> dp(m+1,vector<int>(n+1,INT_MAX));dp[0][1]=dp[1][0]=0;//保证第一个格子起始位置的合理性for(int i=1;i<=m;i++){for(int j=1;j<=n;j++){dp[i][j]=min(dp[i-1][j],dp[i][j-1])+grid[i-1][j-1];//注意下标的映射}}return dp[m][n]; }
};
3.地下城游戏
链接: 174. 地下城游戏
恶魔们抓住了公主并将她关在了地下城 dungeon 的 右下角 。地下城是由 m x n 个房间组成的二维网格。我们英勇的骑士最初被安置在 左上角 的房间里,他必须穿过地下城并通过对抗恶魔来拯救公主。
骑士的初始健康点数为一个正整数。如果他的健康点数在某一时刻降至 0 或以下,他会立即死亡。
有些房间由恶魔守卫,因此骑士在进入这些房间时会失去健康点数(若房间里的值为负整数,则表示骑士将损失健康点数);其他房间要么是空的(房间里的值为 0),要么包含增加骑士健康点数的魔法球(若房间里的值为正整数,则表示骑士将增加健康点数)。
为了尽快解救公主,骑士决定每次只 向右 或 向下 移动一步。
返回确保骑士能够拯救到公主所需的最低初始健康点数。
注意:任何房间都可能对骑士的健康点数造成威胁,也可能增加骑士的健康点数,包括骑士进入的左上角房间以及公主被监禁的右下角房间。

题解
骑士要从左上角出发到右下角救公主。
- 骑士的初始健康点数为一个正整数。
- 如果他的健康点数在某一时刻降至 0 或以下,他会立即死亡。
- 骑士每次只能向左向右移动。骑士每到一个格子要么就是减去健康点、要么加上健康点、要么不增不加。
注意:任何房间都可能对骑士的健康点数造成威胁,也可能增加骑士的健康点数
- 包括骑士进入的左上角房间,以及公主被监禁的右下角房间。(虚拟化)
- 也就是说到达右下角救公主也可能减去健康点。
- 如下面例子,第一个格子是-2,我们可以给个3 ,但是走到-3,骑士就死了。
- 我们也就知道给3不行,然后就给个6,走到右下角骑士也死了也是不行的。
因此最终给个7才真正救出公主。

- 可以抽象理解为,我就是要走一个得结果为一个最小负数的路径,这样求得是他至少需要的健康点数
1.状态表示
- 经验 + 题目要求
经验我们有两个,1.以某个位置为结尾,巴拉巴拉。 2.以某个位置为起点,巴拉巴拉。
- 以某个位置为结尾,巴拉巴拉
- dp[i][j] 表示:从起点出发,到达[i][j]位置的时候,所需要的最低初始健康点数。
但是回到刚才的分析,当往下走的时候,发现骑士死了,才知道初始给的健康点数是不对。
还需要重新给。
- 显然是不行的。因此以某个位置为结尾这种经验定状态标识是不行的,所以换一种。
- 以某个位置为起点,巴拉巴拉
- dp[i][j] 表示:从[i][j] 为起点,到达终点,所需最低初始健康值
2.状态转移方程
- 还是根据最近的一步,划分问题
以[i][j]位置起点,下一步可以向下向右走。那走到下一步的时候是不是还要保证下一步还可以往下在走啊。
- 假设健康值是x ,然后往右走
- 是不是要保证 x + d[i][j] >= dp[i][j+1] ----> x >= dp[i][j+1] - d[i][j]
- 又因为所需的是最低初始健康值,因此往右走仅需 x = dp[i][j+1] - d[i][j],同理往下走 x = dp[i+1][j] - d[i][j]
又因为所需最低初始健康值所以取它们中最小值。

- 其实这里还可以这样理解,dp[i][j]表示 从[i][j]位置出发,到达终点,所需最低初始健康值。
- 每次只能向右向下走,[i][j]最近一步 [i][j+1] 和 [i+1][j],那如果知道 dp[i][j+1]和dp[i+1][j] 到达终点,所需最低初始健康值,取它俩中最小,在减去 d[i][j] 到达终点,所需最低初始健康值了吗。
- dp[i][j] =min(dp[i][j+1],dp[i+1][j]) -d[i][j]
但是这里还有一个非常细的细节,如果dp[i][j] 等于一个负数怎么办。
- 也就是说d[i][j] 这里是一个很大的血包,dp[i][j] 为负值 难道要一个死去的骑士要去救公主吗
- 显示是不合常识的,因此要处理一下。
- 这个血包就足以骑士走接下来的路,因此给骑士一个最低的健康点 1 就行了。
- dp[i][j]=max(1,dp[i][j]);
- 确保 骑士的血量 不可为负数
3.初始化
多给一行一列
- 不过多给一行一列是给在下面的。
- 所以 下表映射 没有改变~
只关注,虚拟节点的值,要保证接下来填表正确。
- 骑士走到右下角救到公主之后,向右向下走还必须保证骑士活着健康点还必须是最低,因此dp[m][n-1] = dp[m-1][n] = 1
- 其他虚拟节点的值为了不影响dp[i][j]找 min,直接给+∞。

4.填表顺序
- 从 右下角 往 左上角
5.返回值
- dp[0][0]
class Solution {
public:int calculateMinimumHP(vector<vector<int>>& dungeon) {//从 i,j 出发,到m,n的最小生命值int m=dungeon.size(),n=dungeon[0].size();vector<vector<int>> dp(m+1,vector<int>(n+1,INT_MAX));//min dp[m-1][n]=dp[m][n-1]=1;//要能 活着救公主for(int i=m-1;i>=0;i--){for(int j=n-1;j>=0;j--){dp[i][j]=min(dp[i+1][j],dp[i][j+1])-dungeon[i][j]; //-去 扣血,就是生命值dp[i][j]=max(1,dp[i][j]);//不为负哦}}return dp[0][0];//从 0,0 出发~}
};相关文章:
[dp4_路径问题] 下降路径最小和 | 最小路径和 | 地下城游戏
目录 1.下降路径最小和 题解 2.最小路径和 题解 3.地下城游戏 题解 做算法题的时候,谨记图画得越详细越好,思路想的越清晰越好,然后再用代码实现一下就好啦 1.下降路径最小和 链接:931. 下降路径最小和 给你一个 n x n 的…...
EasyExcel 数据字典转换器实战:注解驱动设计
一、场景痛点与解决方案 1. 问题背景 在 Excel 导入导出场景中,开发者常面临以下问题: 数据可读性差:数据库存储的字典值(如 1、true)直接导出时难以理解双向转换复杂:导入时需将用户输入的标签反向解析…...
【蓝桥杯】算法笔记2
这篇文章主要记录动态规划方面的学习。 动态规划的核心思想: 把大问题分解成小问题,记住小问题的解,避免重复计算。 动态规划(DP)的三大特点: ①最优子结构:大问题的最优解可以由小问题的最优解推导出来 ②重叠子问题:在求解过程中会反复遇到相同的小问题 ③无后效…...
解决STM32CubeMX中文注释乱码
本人采用【修改系统环境变量】的方法 1. 使用快捷键 win X,打开【系统R】,点击【高级系统设置】 2. 点击【环境变量】 3. 点击【新建】 4.按图中输入【JAVA_TOOL_OPTIONS】和【-Dfile.encodingUTF-8】,新建环境变量后重启CubeMX即可。 解释…...
AI产品的上层建筑:提示词工程、RAG与Agent
上节课我们拆解了 AI 产品的基础设施建设,这节课我们聊聊上层建筑。这部分是产品经理日常工作的重头戏,包含提示词、RAG 和 Agent 构建。 用 AI 客服产品举例,这三者的作用是这样的: 提示词能让客服很有礼貌。比如它会说&#x…...
基于自定义注解+反射+AOP+Redis的通用开关设计:在投行交易与风控系统的落地实践
一句话总结🤣 一个注解让业务逻辑自动切换,Redis当起了隐形操盘手 业务痛点和需求场景 交易系统需支持毫秒级动态切换报价策略,如切换到备用流动性通道风控模型需支持灰度发布(10%流量测试新权重算法)和紧急熔断&am…...
RK3588使用笔记:ubuntu/麒麟系统功能测试程序
一、前言 本编文章记录在使用嵌入式系统中的一些功能测试demo程序,大部分都是AI写的,哈哈哈,确实很有帮助,但是得根据自身设备实际情况和知道如何问AI,才能得出你想要的结果,本文就记录一些ubuntu/麒麟系统…...
Unity中优化绘制调用整理
DrawCall 指的是 CPU 向 GPU 发送渲染指令的过程,在 Unity 中,每次渲染一个网格时,CPU 都需要向 GPU 发送一系列的渲染指令,这个过程被称为一次绘制调用(Draw Call)。 1.GPU实例化 使用: 2.绘…...
ubuntu开启黑屏现象解决
文章目录 前言一、问题描述二、解决方案1. 检查显卡驱动解决步骤: 2. 修复 GRUB 配置解决步骤: 3. 使用恢复模式解决步骤: 三、验证与总结 前言 在使用 Ubuntu 操作系统时,一些用户可能会遇到开机后屏幕黑屏的现象。这种问题可能…...
深度学习deeplearn3
# Jupyter Notebook魔法命令,用于在Notebook中内联显示图表 %matplotlib inline# 导入NumPy库,用于高效的数值计算 import numpy as np# 从matplotlib_inline库导入backend_inline模块,用于设置图表显示格式 from matplotlib_inline import b…...
Mac强制解锁APP或文件夹
当Mac安装过火绒企业版、云安全访问服务之类的APP需要卸载的时候,会发现需要管理员密码,正常的卸载流程走不下去,直接删除APP,会提示“不能完成此操作,xxx已锁定”的信息,此处就记录一下如何关闭锁定状态&a…...
android开发:zxing-android-embedded竖屏扫描功能
Android 点击按钮调用竖屏二维码扫描 提示:zxing-android-embedded插件已过时,建议更换别的。 场景:Home页面上有个扫描按钮,点击后打开摄像头完成扫描功能,扫描时要求竖屏。 方案:使用zxing-android-embe…...
SQL语句(二)—— DML
目录 一、添加数据 1、给指定字段添加数据 2、给全部字段添加数据 3、批量添加数据 二、修改数据 1、修改数据的具体语法 2、案例分析 3、注意事项 三、删除数据 1、删除数据的具体语法 2、案例 3、注意事项 DML全称是Data Manipulation Language,即数据…...
2.2 路径问题专题:LeetCode 63. 不同路径 II
动态规划解决LeetCode 63题:不同路径 II(含障碍物) 1. 题目链接 LeetCode 63. 不同路径 II 2. 题目描述 一个机器人位于 m x n 网格的左上角,每次只能向右或向下移动一步。网格中可能存在障碍物(标记为 1ÿ…...
Linux系统程序设计:从入门到高级Day02
这一篇 我带大家复习一下,C语言中的文件 那一部分 大家注意 这里的图并非原创 是当时我老师的图片 本片作用主要是 后续会有文件相关操作,这篇帮大家复习C语言文件中的内容 有助于大家后面的理解。 文章中代码大多是图片格式,是因为这是我…...
2025高频面试设计模型总结篇
文章目录 设计模型概念单例模式工厂模式策略模式责任链模式 设计模型概念 设计模式是前人总结的软件设计经验和解决问题的最佳方案,它们为我们提供了一套可复用、易维护、可扩展的设计思路。 (1)定义: 设计模式是一套经过验证的…...
【LeetCode 热题100】208:实现 Trie (前缀树)(详细解析)(Go语言版)
🚀 力扣热题 208:实现 Trie (前缀树)(详细解析) 📌 题目描述 力扣 208. 实现 Trie (前缀树) Trie(发音类似 “try”)是一种树形数据结构,用于高效地存储和检索字符串集合中的键。实…...
CSS 父类元素的伪类 选择器
父元素的 :hover 状态可以影响子元素的样式。当父元素处于 :hover 状态时,可以通过 CSS 的选择器为子元素设置样式。 .parent:hover .child 这种选择器叫做 后代选择器(Descendant Selector) ,结合了 :hover 伪类。它的作用是&…...
目前来讲 有哪些三维重建算法,哪个算法效果好
三维重建是计算机视觉和图形学的重要研究方向,其算法在不同场景下的效果差异较大。以下是当前主流的三维重建算法及其特点,按技术路线分类整理: 1. 传统几何方法 (1)结构光(Structured Light…...
快速掌握MCP——Spring AI MCP包教包会
最近几个月AI的发展非常快,各种大模型、智能体、AI名词和技术和框架层出不穷,作为一个业余小红书博主的我最近总刷到MCP这个关键字,看着有点高级我也来学习一下。 1.SpringAI与functionCall简单回顾 前几个月我曾写过两篇关于SpringAI的基础…...
KUKA机器人查看运行日志的方法
对于KUKA机器人的运行日志都是可以查看和导出的,方便查找问题。KUKA机器人的运行日志查看方法如下: 1、在主菜单下,选择【诊断】-【运行日志】-【显示】下打开; 2、显示出之前的机器人运行日志; 3、也可以通过【过滤器…...
MySQL 基础使用指南-MySQL登录与远程登录
MySQL 基础使用指南 1. 登录 MySQL 数据库的命令解析 命令格式: mysql -u用户名 -p密码参数说明: -u(user 的缩写):指定登录用户。例如 -uroot 表示以 root 用户登录。-p(password 的缩写)&a…...
web-ui windows安装与配置
web-ui windows安装与配置 安装然后安装依赖 运行配置 安装 git clone https://github.com/browser-use/web-ui.git先把clone下来 需要有python环境 最好是 Python 3.11 这里就不赘述了 然后安装依赖 pip install -r requirements.txt运行 python webui.py --ip 127.0.0.1 …...
游戏引擎学习第201天
仓库:https://gitee.com/mrxiao_com/2d_game_5 回顾之前的内容,并遇到了一次一阶异常(First-Chance Exception)。 欢迎来到新一期的开发过程,我们目前正在编写调试接口代码。 当前,我们已经在布局系统上进行了一些工…...
Doris:打破 SQL 方言壁垒,构建统一数据查询生态
在大数据领域,不同的数据库系统往往使用不同的 SQL 方言。这就好比不同地区的人说着不同的语言,给数据分析师和开发人员带来极大的困扰。当企业需要整合多个数据源进行分析时,可能要花费大量时间和精力,在不同的 SQL 语法之间切换…...
github合并多个commit message以及rebase解决文件冲突
深度学习求解PDE相关代码全部在我的仓库添加链接描述,自取 github仓库合并多个commit message 问题描述如下: 第一步:确保自己在对应分支上 比如说现在我要合并issue/108分支的提交记录,使用git log --oneline查看提交记录一…...
【零基础入门unity游戏开发——2D篇】SortingGroup(排序分组)组件
考虑到每个人基础可能不一样,且并不是所有人都有同时做2D、3D开发的需求,所以我把 【零基础入门unity游戏开发】 分为成了C#篇、unity通用篇、unity3D篇、unity2D篇。 【C#篇】:主要讲解C#的基础语法,包括变量、数据类型、运算符、…...
系统与网络安全------Windows系统安全(5)
资料整理于网络资料、书本资料、AI,仅供个人学习参考。 磁盘分区管理 磁盘的分区管理 WinR运行,执行“diskmgmt.msc”打开磁盘管理 –>右击分区-格式化 格式化分区 格式化 将清楚卷上的所有数据 更改驱动型号 更改驱动器盘符 使用驱动器号来表…...
springboot—— Shiro实现认证和授权功能
一、数据库模板设计 在本文中,我们使用RBAC(Role-Based Access Control,基于角色的访问控制)模型设计用户,角色和权限间的关系。简单地说,一个用户拥有若干角色,每一个角色拥有若干权限。这样&a…...
牛客 除2问题
除2! 贪心堆 让偶数入堆 注意点: 1.判断堆是否为空再进行操作 2. 为了防止超时,我们采取先求和的方式,后面调整之后再减掉,可以节省一次遍历的时间。 3.注意数据范围,要用long long #include<iost…...
