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

目录

引言:

例题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://xiahunao.cn/news/2812952.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

【正式成立】工业互联网甄选联盟

工业互联网甄选联盟 联盟文件&#xff08;PDF&#xff09;&#xff1a;下载链接&#xff0c;提取码&#xff1a;m5d7 依托将近20年工业领域的智能制造相关项目实施经验、管理经验和产品开发经验&#xff1b;依托iNeuOS工业互联网操作系统、人工智能物流系统、MES制造执行系统等…

动态规划-最长公共子串(c)

动态规划 动态规划&#xff08;dynamic programming&#xff09;是一种算法设计方法。基本思想是在对一个问题的多阶段决策中&#xff0c;按照某一顺序&#xff0c;根据每一步所选决策的不同&#xff0c;会引起状态的转移&#xff0c;最后会在变化的状态中获取到一个决策序列。…

pdf如何压缩文件大小?pdf文件在线压缩方法介绍

在日常工作中&#xff0c;我们经常使用PDF文件进行传输和保存&#xff0c;然而&#xff0c;有时候我们会遇到过大的PDF文件&#xff0c;这不仅会导致传输困难&#xff0c;还会占用过多的设备空间&#xff0c;因此&#xff0c;我们需要对PDF压缩一下以便更轻松地传输和保存&…

2024年【安全员-B证】考试资料及安全员-B证模拟考试

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 安全员-B证考试资料是安全生产模拟考试一点通总题库中生成的一套安全员-B证模拟考试&#xff0c;安全生产模拟考试一点通上安全员-B证作业手机同步练习。2024年【安全员-B证】考试资料及安全员-B证模拟考试 1、【多选…

WebFilter【通过过滤器实现登录判断】

WebFilter【通过过滤器实现登录判断】 /*** 检查用户是否已经完成登录*/ WebFilter(filterName "loginCheckFilter",urlPatterns "/*") Slf4j public class LoginCheckFilter implements Filter{//路径匹配器&#xff0c;支持通配符public static final …

MATLAB环境下基于变分贝叶斯的组织学病理图像颜色盲反卷积方法

图像盲反卷积问题仅根据模糊图像估计清晰图像和模糊核&#xff0c;也是一个欠定问题且求解更加困难。但图像盲反卷积算法更实际&#xff0c;因为许多情况下&#xff0c;模糊核都是未知或部分已知的。求解盲反卷积问题需要为未知量选择适当的先验模型&#xff0c;以得到清晰图像…

SM100-40-080-P0-45-S1-B电机厦门参详

SM100-40-080-P0-45-S1-B交流伺服电机也是无刷电机&#xff0c;分为同步和异步电机&#xff0c;运动控制中一般都用同步电机&#xff0c;它的功率范围大&#xff0c;可以做到很大的功率。大惯量&#xff0c;最高转动速度低&#xff0c;且随着功率增大而快速降低。因而适合做低速…

JSON简介以及如何在Python中使用JSON

什么是JSON&#xff1f; JSON是"JavaScript Object Notation"的简称&#xff0c;是一种数据交换格式 JSON格式 假设我们有一个对象&#xff0c;这个对象有两个属性&#xff1a;“name”跟“age”。 在JSON中是这样表达的&#xff1a; { "name":"男孩…

APP软件设计要注意的问题

设计一个成功的APP软件需要考虑多个方面&#xff0c;成功的APP设计需要综合考虑用户体验、性能稳定性、安全性、兼容性、反馈机制、内容质量和法律合规等多个方面&#xff0c;不断优化和改进以满足用户需求并提升用户满意度。以下是一些需要注意的问题&#xff0c;希望对大家有…

免费的通配符(泛域名证书)?

通配符证书&#xff08;Wildcard SSL Certificate&#xff09;是一种SSL证书&#xff0c;它可以用于保护一个域名及其所有的子域名。与传统的SSL证书不同&#xff0c;传统SSL证书仅用于保护一个单独的完全限定域名或一个子域名。通配符证书通过在域名前加上一个星号&#xff08…

VUE从0到1创建项目及基本路由、页面配置

一、创建项目:(前提已经安装好vue和npm) 目录:E:\personal\project_pro\ windows下,win+R 输入cmd进入命令行: cd E:\personal\project_pro E:# 创建名为test的项目 vue create test# 用上下键选择vue2或vue3,回车确认创建本次选择VUE3 创建好项目后,使用…

【可实战】被测系统业务架构、系统架构、技术架构、数据流、业务逻辑分析

一、为什么要学习 更深的理解业务逻辑&#xff08;公司是做什么的&#xff1f;它最重要的商务决策是什么&#xff1f;它里面的数据流是怎么做的&#xff1f;有哪些业务场景&#xff1f;考验你对这家公司、对所负责业务的熟悉程度。公司背后服务器用什么软件搭建的&#xff1f;…

面试官: 反射了解么?

目录 反射 什么是反射 ? 获取Class对象的四种方式 反射相关API 类对象常用API Filed常用API Method常用API Constructors常用API 反射的使用场景? 反射的实现原理 ?(todo) 反射为什么这么慢 ? 反射的优缺点 反射中&#xff0c;Class.forName和ClassLoader的区别…

【前沿热点视觉算法|Sora|GPT4相关】-显著目标检测的深度增强交叉模态级联网络

计算机视觉算法分享。问题或建议&#xff0c;请文章私信或者文章末尾扫码加微信留言。sora 具体介绍和使用方法&#xff1a;OpenAI Sora 下一代生产力&#xff1a;最新小白必看教程 | 解剖Sora的前世今生 | Sora核心源码目前 openai 官方还未开放 sora 灰度&#xff0c;不过根据…

Linux学习笔记:fork()函数

TOC fork()函数的作用是什么? fork函数一般是用来创建进程的,在fork函数执行后,如果成功创建新进程就会出现两个进程,一个是父进程,一个是子进程 就像火影忍者中的分身术一样,fork之后就会进程就会分出和他一样的分身出来 fork()函数怎么使用? 在使用fork函数之前需要加上…

腾讯云4核8G服务器收费贵不贵?

腾讯云4核8G服务器多少钱&#xff1f;轻量应用服务器4核8G12M带宽一年446元、646元15个月&#xff0c;云服务器CVM标准型S5实例4核8G配置价格15个月1437.3元&#xff0c;5年6490.44元&#xff0c;标准型SA2服务器1444.8元一年&#xff0c;在txy.wiki可以查询详细配置和精准报价…

kubeadm部署K8S

部署二主二从 主1&#xff1a;192.168.116.17 主2&#xff1a;192.168.116.18 从1&#xff1a;192.168.116.12 从2&#xff1a;192.168.116.13 注意事项&#xff1a; master节点的cpu核心数必须要求大于2 K8S最新版本并不一定是最好的&#xff0c;相对于旧版本&#xff…

RISC-V SoC + AI | 在全志 D1「哪吒」开发板上,跑个 ncnn 神经网络推理框架的 demo

引言 D1 是全志科技首款基于 RISC-V 指令集的 SoC&#xff0c;主核是来自阿里平头哥的 64 位的 玄铁 C906。「哪吒」开发板 是全志在线基于全志科技 D1 芯片定制的 AIoT 开发板&#xff0c;是目前还比较罕见的使用 RISC-V SoC 且可运行 GNU/Linux 操作系统的可量产开发板。 n…

配置用户通过IPv6方式上网

组网需求 运营商为企业分配了WAN侧的IPv6地址1111:2222:A0EE:6::2/64和LAN侧的IPv6地址1111:3333:E840:2::1/64&#xff0c;企业通过运营商提供的IPv6地址配置上网。 图1 配置用户通过IPv6方式上网 操作步骤 1、在IPS上的配置 interface GigabitEthernet0/0/4 ipv6 enable…

【视频编码\VVC】量化基础知识

量化&#xff1a;是将信号的连续取值&#xff08;大量离散取值&#xff09;映射为有限多个离散赋值的过程。实现信号取值多对一的映射。可以有效减少信号取值的空间&#xff0c;进而获得更好的压缩效果。 根据输出和输入数据的类型&#xff0c;可以将量化器分为标量量化SQ和矢…