【动态规划】路径问题 {二维动态规划;选择合适的状态表示方法;创建虚拟节点}

本文主要是介绍【动态规划】路径问题 {二维动态规划;选择合适的状态表示方法;创建虚拟节点},希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、经验总结

选择合适的状态表示方法
一般的,状态表示的方法有两种:

  1. 以[i, j]位置为终点,正向填表;用之前的状态推导出dp[i][j]的值(从哪里来);
  2. 以[i, j]位置为起点,反向填表;用之后的状态推导出dp[i][j]的值(到哪里去);

创建虚拟节点

  • 一般虚拟节点设置在需要特殊处理的边界位置,在二维动态规划中,可以出现在矩阵的四边。
  • 需要根据题目要求设置虚拟节点的初始值,以确保后续填表的正确性。
  • 注意dp表与原始表的下标映射关系。

二、相关编程题

2.1 不同路径

题目链接

62. 不同路径 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int uniquePaths(int m, int n) {vector<vector<int>> dp(m+1, vector<int>(n+1, 0)); //多开一行一列作为虚拟节点dp[0][1] = 1; //虚拟节点的初始值,保证后续填表的正确性for(int i = 1; i <= m; ++i) //从上往下{for(int j = 1; j <= n; ++j) //从左往右{dp[i][j] = dp[i-1][j] + dp[i][j-1];}}return dp[m][n];}
};

2.2 不同路径Ⅱ

题目链接

63. 不同路径 II - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int uniquePathsWithObstacles(vector<vector<int>>& obs) {int m = obs.size(), n = obs[0].size();vector<vector<int>> dp(m+1, vector<int>(n+1, 0));dp[0][1] = 1;for(int i = 1; i <= m; ++i){for(int j = 1; j <= n; ++j){if(obs[i-1][j-1] == 1) continue; //注意dp表与原矩阵下标的映射关系dp[i][j] = dp[i-1][j] + dp[i][j-1];}}return dp[m][n];}
};

2.3 珠宝的最高价值

题目链接

LCR 166. 珠宝的最高价值 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int jewelleryValue(vector<vector<int>>& frame) {int m = frame.size(), n = frame[0].size();vector<vector<int>> dp(m+1, vector<int>(n+1));for(int i = 1; i <= m; ++i){for(int j = 1; j <= n; ++j){dp[i][j] = max(dp[i][j-1], dp[i-1][j])+frame[i-1][j-1];}}return dp[m][n];}
};

2.4 下降路径最小和

题目链接

931. 下降路径最小和 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int minFallingPathSum(vector<vector<int>>& matrix) {int n = matrix.size();vector<vector<int>> dp(n+1, vector<int>(n+2, INT_MAX)); //dp表多创建1行2列for(int j = 0; j < n+2; ++j) //将第一行(虚拟节点)初始化为0,两边的两列初始化为INT_MAX{dp[0][j] = 0;}auto min = [](int a, int b, int c)->int{int tmp = a<b? a:b;return c<tmp? c:tmp;};for(int i = 1; i <= n; ++i) //从上往下填表{for(int j = 1; j <= n; ++j){dp[i][j] = min(dp[i-1][j-1], dp[i-1][j], dp[i-1][j+1])+matrix[i-1][j-1];}}int ret = INT_MAX; //选出最后一行的最小值作为结果for(int j = 1; j <= n; ++j){if(ret>dp[n][j]){ret = dp[n][j];}}return ret;}};

2.5 最小路径和

题目链接

64. 最小路径和 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

编写代码

class Solution {
public:int minPathSum(vector<vector<int>>& grid) {int m = grid.size(), n = grid[0].size();vector<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] = grid[i-1][j-1] + min(dp[i-1][j], dp[i][j-1]);}}return dp[m][n];}
};

2.6 地下城游戏

题目链接

174. 地下城游戏 - 力扣(LeetCode)

题目描述

在这里插入图片描述

算法原理

在这里插入图片描述

本题的难点在于怎么处理血量增加的问题, 增加血量不能为之前的损失提供帮助,只会对后续有帮助。
这意味着正向填表是困难的,但是反向填表很好做。以当前位置为起点,从后往前推,当前如果可以治愈,那么当前的最低血量就是下一关的最低血量减去治疗量,注意不可以<1。

编写代码

class Solution {
public:int calculateMinimumHP(vector<vector<int>>& dun) {int m = dun.size(), n = dun[0].size();vector<vector<int>> dp(m+1, vector<int>(n+1, INT_MAX));dp[m][n-1] = dp[m-1][n] = 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]) - dun[i][j];dp[i][j] = max(1, dp[i][j]);}}return dp[0][0];}
};

这篇关于【动态规划】路径问题 {二维动态规划;选择合适的状态表示方法;创建虚拟节点}的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1080911

相关文章

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

Vue3绑定props默认值问题

《Vue3绑定props默认值问题》使用Vue3的defineProps配合TypeScript的interface定义props类型,并通过withDefaults设置默认值,使组件能安全访问传入的... 目录前言步骤步骤1:使用 defineProps 定义 Props步骤2:设置默认值总结前言使用T

JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法

《JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法》:本文主要介绍JavaScript中比较两个数组是否有相同元素(交集)的三种常用方法,每种方法结合实例代码给大家介绍的非常... 目录引言:为什么"相等"判断如此重要?方法1:使用some()+includes()(适合小数组)方法2

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

SpringBoot路径映射配置的实现步骤

《SpringBoot路径映射配置的实现步骤》本文介绍了如何在SpringBoot项目中配置路径映射,使得除static目录外的资源可被访问,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一... 目录SpringBoot路径映射补:springboot 配置虚拟路径映射 @RequestMapp

504 Gateway Timeout网关超时的根源及完美解决方法

《504GatewayTimeout网关超时的根源及完美解决方法》在日常开发和运维过程中,504GatewayTimeout错误是常见的网络问题之一,尤其是在使用反向代理(如Nginx)或... 目录引言为什么会出现 504 错误?1. 探索 504 Gateway Timeout 错误的根源 1.1 后端

Web服务器-Nginx-高并发问题

《Web服务器-Nginx-高并发问题》Nginx通过事件驱动、I/O多路复用和异步非阻塞技术高效处理高并发,结合动静分离和限流策略,提升性能与稳定性... 目录前言一、架构1. 原生多进程架构2. 事件驱动模型3. IO多路复用4. 异步非阻塞 I/O5. Nginx高并发配置实战二、动静分离1. 职责2