【动态规划】【路径问题】下降路经最小和、最小路径和、地下城游戏

server/2024/10/25 4:26:17/

4. 下降路径最小和

931. 下降路径最小和
image.png|548

算法原理

  1. 确定状态表示

    • dp[i][j] 表示:到达 [i, j] 位置,最小的下降路径
  2. 状态转移方程

  • dp[i][j]

    • [i-1, j-1] 到达 [i, j] ==> dp[i-1][j-1] + m[i][j]
    • [i-1, j] 到达 [i, j] ==> dp[i-1][j] + m[i][j]
    • [i-1, i+1] 到达 [i, j] ==> dp[i-1][j+1] + m[i][j]
  • dp[i][j] = min(上面三个) + m[i][j]

  1. 初始化
    • 目的是为了让我们再填表的过程中不会出现越界的情况

里面的值,要保证后面的填表是正确的
image.png

  • 绿星的地方都可能会越界
  • 进行绿框范围的虚拟节点构造

  • 虚拟出的第一行全部填 0,就可以保证原表的第一行都是 0
  • 但从原表的第二行开始,每个格子都是取前三者之间的最小值,所以下面虚拟的节点就不能填最小的值 0 了,不然每个格子都是 0。所以都取正无穷大

下标的映射

  • 整个表向右下移动了一个单位长度
  • (0, 0)——>(1, 1)

在初始化的时候,可以把所有虚拟出的节点都设为 +∞,然后将第一行改为 0 就可以了

  1. 填表顺序

    • 从上往下
  2. 返回值

    • 这里不是返回 dp[m][n] 的值
    • 返回 dp 表中最行一行的最小值

代码编写

public int minFallingPathSum(int[][] matrix) {  //1. 创建 dp 表  int n = matrix.length;  int[][] dp = new int[n+1][n+2];  //2. 初始化  for (int i = 1; i <= n; i++) {  //第一列和最后一列  dp[i][0] = dp[i][n+1] = Integer.MAX_VALUE;  }  //3. 填表  for (int i = 1; i <= n; i++) {  for (int j = 1; j <= n; j++) {  dp[i][j] = Math.min(dp[i-1][j], Math.min(dp[i-1][j], dp[i-1][j+1])) + matrix[i-1][j-1];  }  }  int ret = Integer.MAX_VALUE;  for (int i = 1; i <= n; i++) {  ret = Math.min(ret, dp[n][i]);  }  return ret;  
}
  • 取最值只能两个进行比较
  • 注意无穷值的写法

时间复杂度:n*n(两个 for 循环)
空间复杂度:n*n(弄了个二维 dp 表)

5. 最小路径和

64. 最小路径和
image.png|589

算法原理

  1. 确定状态表示

    • dp[i][j] 表示:到达 [i, j] 位置时,最小路径和
  2. 状态转移方程

  • dp[i][j]
    • [i-1, j] 走过来==> dp[i-1][j] + g[i][j]
    • [i, j-1] 走过来==> dp[i][j-1] + g[i][j]
    • `dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + g[i][j]
  1. 初始化
  • 因为是取最小值,所以虚拟的节点要小,以免被误取
  • 但起点需要是 0,所以它左边和上面的虚拟节点 dp[0][1]dp[1][0] 需要是 0
  1. 填表顺序

    • 从上往下
    • 从左往右
  2. 返回值

    • 返回 dp[m][n]

代码编写

public int minPathSum(int[][] grid) {  //1. 创建 dp 表  int m = grid.length;  int n = grid[0].length;  int[][] dp = new int[m+1][n+1];  //2. 初始化  for (int i = 0; i <= n; i++)  dp[0][i] = Integer.MAX_VALUE;  for (int i = 0; i <= m; i++)   dp[i][0] = Integer.MAX_VALUE;  dp[0][1] = dp[1][0] = 0;  //3. 填表  for (int i = 1; i <= m; i++) {  for (int j = 1; j <= n; j++) {  dp[i][j] = Math.max(dp[i-1][j], dp[i][j-1]) + grid[i-1][j-1];  }  }  return dp[m][n];  
}

6. 地下城游戏

174. 地下城游戏
image.png|508

算法原理

  1. 确定状态表示
    • dp[i][j] 表示:从 [i, j] 位置出发,到达终点,所需的最低初始健康点数

这里不能以 [i, j] 为终点构建状态表示,

  1. 状态转移方程
  • dp[i][j],此时的点数必须要>=下一个要到的地方 dp 值
    • 从此处往右走,x+d[i][j] >= dp[i][j+1],所以 x >= dp[i][j+1] - d[i][j] 即可
    • 从此处往下走,同理 x >= d[i+1][j] - d[i][j] 即可

如果 d[i][j] 太大,就是说在那一格有个很大的血包。减完之后就变成一个负值了(你是一个负血的状态,通过这个格子之后也能顺利通过),这是不符合逻辑的。

  • 所以我们要把 dp[i][j]1 放在一起取一下 max
  • 如果算出来是负数,就更新为 1
  • 如果是大于等于 1 的数,就保持
  1. 初始化
    image.png|480

我们关注的是格子的下面和右边的状态,所以可能会越界的是最下面一行和最右边一行

  • 我们在最下面和最右边添加辅助节点
  • 此时就不用考虑下标映射关系

里面的值,需要保证后续的填表是正确的

  • 我们看原表终点格,要走出去,终点最少需要 1 点血量
  • 所以只需要把终点下面和右边的格子置为 1 就可以了
  • 其余的位置是两格之间求 min,我们只需要保证辅助的节点不被选上就可以,所以我们将其他的节点设为 +∞
  1. 填表顺序

    • 从下往上
    • 从右往左
  2. 返回值

    • 返回 dp[0][0]

代码编写

public int calculateMinimumHP(int[][] dungeon) {  //1. 创建 dp 表  int m = dungeon.length;  int n = dungeon[0].length;  int[][] dp = new int[m+1][n+1];  //2. 初始化  for (int j = 0; j <= n; j++)  dp[m][j] = Integer.MAX_VALUE;  for (int i = 0; i <= m; i++)  dp[i][n] = Integer.MAX_VALUE;  dp[m][n-1] = dp[m-1][n] = 1;  //3. 填表  for (int i = m-1; i >= 0; i--) {  for (int j = n-1; j >= 0; j--) {  dp[i][j] = Math.min(dp[i][j+1], dp[i+1][j]) - dungeon[i][j];  dp[i][j] = Math.max(dp[i][j], 1);  }  }  return dp[0][0];  
}

http://www.ppmy.cn/server/134608.html

相关文章

【Flutter】基础入门:项目结构

Flutter 是一款用于开发跨平台应用的优秀框架。通过一次编写代码&#xff0c;Flutter 可以将应用部署到 Android、iOS、Web、Windows、Linux 和 macOS 等多个平台。作为 Flutter 开发者&#xff0c;理解 Flutter 项目的目录结构和配置是至关重要的&#xff0c;能够帮助你快速构…

Python科学计算思维导图(Numpy、Matplotlib、Pandas)

Python科学计算思维导图&#xff08;Numpy、Matplotlib、Pandas&#xff09;整理&#xff0c;后期如有 添加或更改会同步更新&#xff0c;敬请关注&#xff01; 整理不易&#xff0c;如有转载&#xff0c;请注明出处&#xff0c;谢谢&#xff01;&#xff01;&#xff01;

Elasticsearch基本使用及介绍

Elasticsearch 1. 关于各种数据库的使用 关于MySQL&#xff1a;是关系型数据库&#xff0c;能清楚的表示数据之间的关系&#xff0c;并且&#xff0c;是基于磁盘存储的&#xff0c;可以使用相对较低的成本存储大量的数据 关于Redis&#xff1a;是基于K-V结构的在内存中读写数…

SpringBoot框架的车辆管理自动化解决方案

2相关技术 2.1 MYSQL数据库 MySQL是一个真正的多用户、多线程SQL数据库服务器。 是基于SQL的客户/服务器模式的关系数据库管理系统&#xff0c;它的有点有有功能强大、使用简单、管理方便、安全可靠性高、运行速度快、多线程、跨平台性、完全网络化、稳定性等&#xff0c;非常…

Soap-UI传参

<?xml version"1.0" encoding"UTF-8"?> <soapenv:Envelope xmlns:soapenv"http://schemas.xmlsoap.org/soap/envelope/" xmlns:web"http://webservice.ihosp.jxmns.thirdparty.his.gocent.com/"><soapenv:Header/>…

perl读取目录,写入文件

perl读取目录&#xff0c;写入文件 此脚本有两个输入参数&#xff0c;第一个参数为需要打印的文件目录&#xff0c;第二个参数为打印后的文件名&#xff1b; 该脚本名称为out_file_full_path #!/bin/perluse 5.010; my $dir $ARGV[0]; # 此为第一个参数&#xff1b; opendi…

显示指定目录下的 .c 文件 Linux环境 C语言实现

问题&#xff1a;显示指定目录下的 .c 文件 算法&#xff1a; 1. opendir ( ) 打开文件夹 2. readdir ( ) 读取文件名 3. 通过字符串比对找出 .c 文件并打印输出 4. closedir ( ) 关闭文件夹 代码&#xff1a; #include<stdio.h> #include<sys/types.h> #includ…

vue,java,webSocket通讯,服务端主动给多客户端发消息

vue在那个页面内&#xff1a; created() {// 可以在created钩子中初始化WebSocket连接this.initWebSocket();}, data: () > {return {webSocket: null, // WebSocket对象}, }, beforeDestroy() {// 组件销毁前关闭WebSocket连接if (this.webSocket) {this.webSocket.close(…