动态规划课堂2-----路径问题

news/发布时间2024/6/16 17:52:27

目录

引言:

例题1:不同路径

例题2:不同路径II

例题3:礼物的最⼤价值

例题4:下降路径最⼩和

例题5:最小路径和

结语:


引言:

在学习完动态规划斐波那契数列模型后,相信大家对动态规划已经有了一定的了解,下面我们继续深入学习动态规划的路径问题,我们一般的解题步骤还是1. 状态表示,2.状态转移方程,3.初始化,4.填表顺序,5.返回值。在写代码时一定要把这5步考虑清楚再写代码,写代码的步骤为1.创建 dp表2.初始化3.填表4.返回值。

例题1:不同路径

链接:不同路径

题目简介:

一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。

机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finish” )。

问总共有多少条不同的路径?

解法(动态规划):

1. 状态表示:

对于这种「路径类」的问题,我们的状态表⽰⼀般有两种形式:

(1)从[i, j] 位置出发到终点有几种路径。

(2)从起始位置出发,到达[i, j] 位置有几种路径。

本题我们选择第二种,dp[i][j] 表示:⾛到[i, j] 位置处,⼀共有多少种⽅式。

2.状态转移方程

简单分析⼀下。如果dp[i][j] 表⽰到达[i, j] 位置的⽅法数,那么到达[i, j] 位置之 前的⼀⼩步,有两种情况:

(1)从[i, j] 位置的上⽅( [i - 1, j] 的位置)向下⾛⼀步,转移到[i, j] 位置

(2)从[i, j] 位置的左⽅( [i, j - 1] 的位置)向右⾛⼀步,转移到[i, j] 位置。

由于我们要求的是有多少种⽅法,因此状态转移⽅程就呼之欲出了: dp[i][j] = dp[i - 1] [j] + dp[i][j - 1] 。

3.初始化

使用辅助结点 。

注意点:(1)辅助结点⾥⾯的值要「保证后续填表是正确的」(2)下标的映射关系。在本题中,「添加⼀⾏」,并且「添加⼀列」后,只需将dp[0][1] 的位置初始化为1 即可。

4.填表顺序

根据「状态转移⽅程」的推导来看,填表的顺序就是「从上往下」填每⼀⾏,在填写每⼀⾏的时候 「从左往右」。

5.返回值

根据「状态表⽰」,我们要返回dp[m][n](添加了一个辅助节点) 的值 。

代码如下:

动态规划编写代码的步骤比较固定,上面那5步是想好思路,下面这4步是编写代码的步骤。

class Solution {public int uniquePaths(int m, int n) {//1.创建dp表//2.初始化//3.填表//4.返回值int[][] dp = new int[m + 1][n + 1];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];}
}

时间复杂度:O(n * m)

空间复杂度:O(n * m)

例题2:不同路径II

链接:不同路径II

题目简介:

一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” )。

机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角(在下图中标记为 “Finish”)。

现在考虑网格中有障碍物。那么从左上角到右下角将会有多少条不同的路径?

网格中的障碍物和空位置分别用 1 和 0 来表示。

 解法(动态规划):

本题是例题一的进阶版,但是两题的解题步骤是类似的不同的地方在于状态转移方程。[i - 1, j] 与[i, j - 1] 位置都是可能有障碍的,此时从上⾯或者左边是不可能到达[i, j] 位置的,也就是说,此时的⽅法数应该是0。由此我们可以得出⼀个结论,只要这个位置上「有障碍物」,那么我们就不需要计算这个位置上的值,直接让它等于0即可。

代码如下:

class Solution {public int uniquePathsWithObstacles(int[][] obstacleGrid) {//1.创建dp表//2.初始化//3.填表//4.返回值int m = obstacleGrid.length;int n = obstacleGrid[0].length;int[][] dp = new int[m + 1][n + 1];dp[0][1] = 1;for(int i = 1;i <= m;i++){for(int j = 1;j <= n;j++){if(obstacleGrid[i - 1][j - 1] == 1){dp[i][j] = 0;}else{dp[i][j] = dp[i - 1][j] + dp[i][j - 1];}}}return dp[m][n];}
}

时间复杂度:O(n * m)

空间复杂度:O(n * m)

例题3:礼物的最⼤价值

链接:礼物的最⼤价值

题目简介:

现有一个记作二维矩阵 frame 的珠宝架,其中 frame[i][j] 为该位置珠宝的价值。拿取珠宝的规则为:

  • 只能从架子的左上角开始拿珠宝
  • 每次可以移动到右侧或下侧的相邻位置
  • 到达珠宝架子的右下角时,停止拿取

注意:珠宝的价值都是大于 0 的。除非这个架子上没有任何珠宝,比如 frame = [[0]]

解法(动态规划):

1. 状态表示:

dp[i][j] 表⽰:⾛到[i, j] 位置处,此时的最⼤价值。

2.状态转移方程

对于dp[i][j] ,我们发现想要到达[i, j] 位置,有两种⽅式:

(1)从[i, j] 位置的上⽅[i - 1, j] 位置,向下⾛⼀步,此时到达[i, j] 位置能 拿到的礼物价值为dp[i - 1][j] + grid[i][j] .

(2)从[i, j] 位置的左边[i, j - 1] 位置,向右⾛⼀步,此时到达[i, j] 位置能 拿到的礼物价值为dp[i][j - 1] + grid[i][j].

最终dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]) + grid[i][j] 。

3.初始化

添加辅助节点

4.填表顺序

填表的顺序是「从上往下填写每⼀⾏」,「每⼀⾏从左往右」。

5.返回值

返回dp[m][n]的值

代码如下:

class Solution {public int jewelleryValue(int[][] frame) {//1.创建dp表//2.初始化//3.填表//4.返回值int m = frame.length;int n = frame[0].length;int[][] dp = new int[m + 1][n + 1];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]) + frame[i - 1][j - 1];}}return dp[m][n];}
}

时间复杂度:O(n * m)

空间复杂度:O(n * m)

例题4:下降路径最⼩和

链接:下降路径最⼩和

题目简介:

给你一个 n x n 的 方形 整数数组 matrix ,请你找出并返回通过 matrix 的下降路径  最小和 。

下降路径 可以从第一行中的任何元素开始,并从每一行中选择一个元素。在下一行选择的元素和当前行所选元素最多相隔一列(即位于正下方或者沿对角线向左或者向右的第一个元素)。具体来说,位置 (row, col) 的下一个元素应当是 (row + 1, col - 1)(row + 1, col) 或者 (row + 1, col + 1) 。

解法(动态规划):

1. 状态表示:

dp[i][j] 表⽰:到达[i, j] 位置时,所有下降路径中的最⼩和。

2.状态转移方程

对于普遍位置[i, j] ,根据题意得,到达[i, j] 位置可能有三种情况:

(1)从正上⽅[i - 1, j] 位置转移到[i, j] 位置.

(2)从左上⽅[i - 1, j - 1] 位置转移到[i, j] 位置;

  (3)   从右上⽅[i - 1, j + 1] 位置转移到[i, j] 位置;

由于我们要的是最小的于是 dp[i][j] = min(dp[i - 1][j], min(dp[i - 1][j - 1], dp[i - 1][j + 1])) + matrix[i][j] 。

3.初始化

辅助节点

在本题中,需要「加上⼀⾏」,并且「加上两列」。所有的位置都初始化为⽆穷⼤,然后将第⼀⾏ 初始化为0即可。

4.填表顺序

表的顺序是从上往下。

5.返回值

返回dp表中最后⼀⾏的最⼩值。

代码如下:

class Solution {public int minFallingPathSum(int[][] matrix) {//1.创建dp表//2.初始化//3.填表//4.返回值int m = matrix.length;//yint n = matrix[0].length;//xint[][] dp = new int[m + 1][n + 2];//辅助节点for(int i = 0;i <= m;i++){dp[i][0] = Integer.MAX_VALUE;dp[i][n + 1] = Integer.MAX_VALUE;}for(int i = 1;i <= n;i++){dp[m][i] = Integer.MAX_VALUE;}for(int i = m - 1;i >= 0;i--){for(int j = 1;j <= n;j++){int min = Math.min(Math.min(dp[i + 1][j - 1],dp[i + 1][j]),dp[i + 1][j + 1]);if(min == Integer.MAX_VALUE){min = 0;}dp[i][j] = matrix[i][j - 1] + min;}}int key = dp[0][1];for(int i = 1;i <= n;i++){if(key > dp[0][i]){key = dp[0][i];}}return key;}
}

时间复杂度:O(n * m)

空间复杂度:O(n * m)

例题5:最小路径和

链接:最⼩路径和

题目简介:

给定一个包含非负整数的 m x n 网格 grid ,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。

说明:每次只能向下或者向右移动一步。

解法(动态规划):

1. 状态表示:

dp[i][j] 表⽰:到达[i, j] 位置处,最⼩路径和是多少。

2.状态转移方程

简单分析⼀下。如果dp[i][j] 表⽰到达到达[i, j] 位置处的最⼩路径和,那么到达 [i, j] 位置之前的⼀⼩步,有两种情况:

(1)从[i - 1, j] 向下⾛⼀步,转移到[i, j] 位置;

(2)从[i, j - 1] 向右⾛⼀步,转移到[i, j] 位置。

由于到[i, j] 位置两种情况,并且我们要找的是最⼩路径,因此只需要这两种情况下的最⼩值,再加上 [i, j] 位置上本⾝的值即可。故最终结果为:dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + grid[i][j]。

3.初始化

辅助节点

在本题中,「添加⼀⾏」,并且「添加⼀列」后,所有位置的值可以初始化为⽆穷⼤,然后让 dp[0][1] = dp[1][0] = 0 即可。如下图所示:

4.填表顺序

「从上往下」填每⼀⾏,每⼀⾏「从左往右」。

5.返回值

返回的结果是dp[m][n] 。

代码如下:

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

时间复杂度:O(n * m)

空间复杂度:O(n * m)

结语:

其实写博客不仅仅是为了教大家,同时这也有利于我巩固自己的知识点,和一个学习的总结,由于作者水平有限,对文章有任何问题的还请指出,接受大家的批评,让我改进,如果大家有所收获的话还请不要吝啬你们的点赞收藏和关注,这可以激励我写出更加优秀的文章。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.bcls.cn/xMsI/8125.shtml

如若内容造成侵权/违法违规/事实不符,请联系编程老四网进行投诉反馈email:xxxxxxxx@qq.com,一经查实,立即删除!

相关文章

redis八股

文章目录 数据类型字符串实现使用场景 List 列表实现使用场景 Hash 哈希实现使用场景 Set 集合实现使用场景 ZSet 有序集合实现使用场景 BitMap实现使用场景 Stream使用场景pubsub为什么不能作为消息队列 数据结构机制SDS 简单动态字符串压缩列表哈希表整数集合跳表quicklistli…

开源现场总线协议栈(ethercat、ethernet/ip、opc ua、profinet、canopen、modbus)

ecat主站及其相关&#xff1a; 1.soem&#xff1a;GitHub - OpenEtherCATsociety/SOEM: Simple Open Source EtherCAT MasterSimple Open Source EtherCAT Master. Contribute to OpenEtherCATsociety/SOEM development by creating an account on GitHub.https://github.com/…

pytest钩子函数-pytest_runtest_logreport提取测试用例相关信息

问题&#xff1a;想在每个日志中记录测试用例开始结束时间&#xff0c;获取到测试用例的名称。 解决办法&#xff1a;使用钩子pytest_runtest_logreport 在pytest中&#xff0c;想要在conftest.py文件中获取正在运行的测试用例的名称&#xff0c;可以使用pytest_runtest_logre…

顺序表的列题(力扣)和旋转数组

文章目录 一.删除有序数组中的重复项&#xff08;取自力扣&#xff09; 二.合并两个有序数组&#xff08;取自力扣&#xff09; 三.旋转数组&#xff08;多解法&#xff09; 前言 见面我们说到了顺序表今天来分享几个有关于顺序表的题目 一.删除有序数组中的重复项&#xff…

性能优化问题思考总结

INP 是什么&#xff1f; Interaction to Next Paint (INP) INP是一项指标&#xff0c;通过观察用户在访问网页期间发生的所有点击、点按和键盘互动的延迟时间&#xff0c;评估网页对用户互动的总体响应情况。 互动是指在同一逻辑用户手势期间触发的一组事件处理脚本。例如&a…

《Large Language Models for Generative Information Extraction: A Survey》阅读笔录

论文地址&#xff1a;Large Language Models for Generative Information Extraction: A Survey 前言 映像中&#xff0c;比较早地使用“大模型“”进行信息抽取的一篇论文是2022年发表的《Unified Structure Generation for Universal Information Extraction》&#xff0c;也…

四、矩阵的分类

目录 1、相等矩阵 2、同形矩阵 3、方阵&#xff1a; 4、负矩阵、上三角矩阵、下三角矩阵&#xff1a; 5、对角矩阵&#xff1a;是方阵 ​编辑7、单位矩阵&#xff1a;常常用 E或I 来表示。它是一个方阵 8、零矩阵&#xff1a; 9、对称矩阵&#xff1a;方阵 1、相等矩阵 …

Mamba详细介绍和RNN、Transformer的架构可视化对比

Transformer体系结构已经成为大型语言模型(llm)成功的主要组成部分。为了进一步改进llm&#xff0c;人们正在研发可能优于Transformer体系结构的新体系结构。其中一种方法是Mamba&#xff08;一种状态空间模型&#xff09;。 Mamba: Linear-Time Sequence Modeling with Select…

mac电脑监控软件哪个好

在Mac电脑使用日益普及的今天&#xff0c;企业对于Mac终端的安全管理需求也日益增长。Mac电脑监控软件作为一种有效的管理工具&#xff0c;能够帮助企业提高数据安全性和员工工作效率。 在众多Mac电脑监控软件中&#xff0c;域智盾软件以其卓越的功能和性能脱颖而出&#xff0c…

数据结构2月25日

第一道&#xff1a; 第二道&#xff1a; 1、插入到prev和next中间 1.new(struct list_head*)malloc(sizeof(struct list_head*)); if(newNULL) { printf("失败\n"); return; } new->nextprev->next; prev->nextnew; return; 2、删除prve和next…

「连载」边缘计算(十九)02-22:边缘部分源码(源码分析篇)

&#xff08;接上篇&#xff09; 从启动函数Start(&#xff09;中可以看到&#xff0c;其以go routine的方式启动很多后台处理服务&#xff0c;具体如下。 1&#xff09;初始化edged的kubeClient&#xff0c;具体如下所示。 // use self defined client to replace fake kube…

EasyRecovery2024个人免费版本电脑手机数据恢复软件下载

EasyRecovery是一款功能强大的数据恢复软件&#xff0c;能够帮助用户恢复丢失、删除、格式化或损坏的数据。无论是由于误操作、病毒攻击、硬盘故障还是其他原因导致的数据丢失&#xff0c;EasyRecovery都能提供有效的解决方案。 该软件支持从各种存储介质恢复数据&#xff0c;…

RV32/64 特权架构 - 特权模式与指令

RV32/64 特权架构 - 特权模式与指令 1 特权模式2 特权指令2.1 mret&#xff08;从机器模式返回到先前的模式&#xff09;2.2 sret&#xff08;从监管模式返回到先前的模式&#xff09;2.3 wfi&#xff08;等待中断&#xff09;2.4 sfence.vma&#xff08;内存屏障&#xff09; …

MySQL - 事务日志

目录 1. redo日志 1.1 为什么需要REDO日志 1.2 REDO日志的好处、特点 1. 好处 2. 特点 1.3 redo的组成 1.4 redo的整体流程 1.5 redo log的刷盘策略 1.6 不同刷盘策略演示 1. 流程图 ​编辑2. 举例 1.7 写入redo log buffer 过程 1.8 redo log file 1. 相关参数…

MCU最小系统电路设计(以STM32F103C8T6为例)

目录 一、何为最小系统&#xff1f; 二、最小系统电路设计 1.电源 &#xff08;1&#xff09;各种名词解释 &#xff08;2&#xff09;为什么会有VDD_1 _2 _3区分&#xff1f; &#xff08;3&#xff09;Mirco USB &#xff08;4&#xff09;5v->3.3v滤波电路 &#…

Linux修改shell工具连接端口

nano /etc/ssh/sshd_config 或者 vi /etc/ssh/sshd_config 或者 vim /etc/ssh/sshd_config

Wireshark过滤DNS协议包语法实战

背景 现网DNS服务器发现CPU突增&#xff0c;发现有可能是客户恶意发起的随机子域名扫描&#xff0c;对服务器进行抓包分析&#xff0c;记录下当时的操作。 抓包 执行命令 tcpdump -iany port 53 and host $ip -nnv -w $ip.pcap进行抓包导出到本地&#xff0c;使用Wireshark进…

【GameFramework框架内置模块】5、下载(Download)

推荐阅读 CSDN主页GitHub开源地址Unity3D插件分享简书地址 大家好&#xff0c;我是佛系工程师☆恬静的小魔龙☆&#xff0c;不定时更新Unity开发技巧&#xff0c;觉得有用记得一键三连哦。 一、前言 【GameFramework框架】系列教程目录&#xff1a; https://blog.csdn.net/q7…

论文阅读《Sylph: A Hypernetwork Framework for Incremental Few-shot Object Detection》

论文地址&#xff1a;https://arxiv.org/abs/2203.13903 代码地址&#xff1a;https://github.com/facebookresearch/sylph-few-shot-detection 目录 1、存在的问题2、算法简介3、算法细节3.1、基础检测器3.2、小样本超网络3.2.1、支持集特征提取3.2.2、代码预测3.2.3、代码聚合…

css4浮动+清除浮动

浮动 一.常见网页布局1.三种布局方式2.布局准则 二.浮动&#xff08;float&#xff09;1.好处2.概念3.三大特性4.使用5.常见网页布局模板6.注意点 三.清除浮动1.why2.本质3.语法4.四种way&#xff08;后三个都是给父级添加&#xff09;清除浮动总结 一.常见网页布局 1.三种布局…
推荐文章