【算法】——动态规划题目讲解

news/2024/5/11 0:21:01

本期继续为大家带来的是关于动态规划类题目的讲解,对于这类题目大家一定要多加练习,争取掌握。

(一)不同路径

链接如下:62. 不同路径

  • 题目如下:

算法思路:

  • 1. 状态表⽰:

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

  • i. 从 [i, j] 位置出发;
  • ii. 从起始位置出发,到达 [i, j] 位置。

这⾥选择第⼆种定义状态表⽰的⽅式:

  • dp[i][j] 表⽰:⾛到 [i, j] 位置处,⼀共有多少种⽅式。

  • 2. 状态转移⽅程:

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

  • i. 从 [i, j] 位置的上⽅( [i - 1, j] 的位置)向下⾛⼀步,转移到 [i, j] 位置;
  • ii. 从 [i, j] 位置的左⽅( [i, j - 1] 的位置)向右⾛⼀步,转移到 [i, j] 位置。

由于我们要求的是有多少种⽅法,因此状态转移⽅程就呼之欲出了:

 💨  dp[i][j] = dp[i - 1] [j] + dp[i][j - 1] 


  • 3. 初始化:

可以在最前⾯加上⼀个「辅助结点」,帮助我们初始化。使⽤这种技巧要注意两个点:

  • i. 辅助结点⾥⾯的值要「保证后续填表是正确的」;
  • ii. 「下标的映射关系」

在本题中,「添加⼀⾏」,并且「添加⼀列」后,只需将 dp[0][1] 的位置初始化为 1 即可。


  • 4. 填表顺序:

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


  • 5. 返回值:

根据「状态表⽰」,我们要返回 dp[m][n] 的值。


代码如下:

class Solution {
public:int uniquePaths(int m, int n) {vector<vector<int>> arr(m+1,vector<int>(n+1,0));arr[0][1] = 1;for(int i=1; i<= m; ++i){for(int j=1; j<= n; j++){arr[i][j] = arr[i-1][j] + arr[i][j-1];}}// 返回结果return arr[m][n];}
};

性能分析:

  • 时间复杂度:O(mn)。

  • 空间复杂度:O(mn)。


(二)不同路径||

链接如下:63. 不同路径 II

题目如下:

 

算法思路:

本题为不同路径的变型,只不过有些地⽅有「障碍物」,只要在「状态转移」上稍加修改就可解决。

  • 1. 状态表⽰:

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

  • i. 从 [i, j] 位置出发;
  • ii. 从起始位置出发,到达 [i, j] 位置。

这⾥选择第⼆种定义状态表⽰的⽅式:

  • dp[i][j] 表⽰:⾛到 [i, j] 位置处,⼀共有多少种⽅式。

  • 2. 状态转移:

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

  • i. 从 [i, j] 位置的上⽅( [i - 1, j] 的位置)向下⾛⼀步,转移到 [i, j] 位置;
  • ii. 从 [i, j] 位置的左⽅( [i, j - 1] 的位置)向右⾛⼀步,转移到 [i, j] 位置。

但是, [i - 1, j] 与 [i, j - 1] 位置都是可能有障碍的,此时从上⾯或者左边是不可能 到达 [i, j] 位置的,也就是说,此时的⽅法数应该是 0。 由此我们可以得出⼀个结论,只要这个位置上「有障碍物」,那么我们就不需要计算这个位置上的值,直接让它等于 0 即可。


  • 3. 初始化:

可以在最前⾯加上⼀个「辅助结点」,帮助我们初始化。使⽤这种技巧要注意两个点:

  • i.    辅助结点⾥⾯的值要「保证后续填表是正确的」;
  • ii. 「下标的映射关系」

在本题中,添加⼀⾏,并且添加⼀列后,只需将 dp[1][0] 的位置初始化为 1 即可。


  • 4. 填表顺序:

根据「状态转移」的推导,填表的顺序就是「从上往下」填每⼀⾏,每⼀⾏「从左往右」。


  • 5. 返回值:

根据「状态表⽰」,我们要返回的结果是 dp[m][n] 。


代码如下:

class Solution {
public:int uniquePathsWithObstacles(vector<vector<int>>& obstacleGrid) {int len = obstacleGrid.size();int widelen=obstacleGrid[0].size();vector<vector<int>> arr(len+1,vector<int>(widelen+1,0));arr[0][1] = 1;for(int i=1; i<= len; ++i){for(int j=1; j<= widelen; ++j){if(obstacleGrid[i - 1][j - 1] == 0)arr[i][j] = arr[i-1][j] + arr[i][j-1];}}return arr[len][widelen];}
};

性能分析:

  • 时间复杂度:O(mn)。

  • 空间复杂度:O(mn)。

 


(三)礼物的最⼤价值

链接如下:剑指 Offer 47. 礼物的最大价值

题目如下:

算法思路:

  • 1. 状态表⽰: 

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

  • i. 从 [i, j] 位置出发,巴拉巴拉;
  • ii. 从起始位置出发,到达 [i, j] 位置,巴拉巴拉。

这⾥选择第⼆种定义状态表⽰的⽅式:

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

  • 2. 状态转移⽅程:

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

  • i. 从 [i, j] 位置的上⽅ [i - 1, j] 位置,向下⾛⼀步,此时到达 [i, j] 位置能 拿到的礼物价值为 dp[i - 1][j] + grid[i][j] ;
  • ii. 从 [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. 初始化:

可以在最前⾯加上⼀个「辅助结点」,帮助我们初始化。使⽤这种技巧要注意两个点:

  • i. 辅助结点⾥⾯的值要「保证后续填表是正确的」;
  • ii. 「下标的映射关系」。

在本题中,「添加⼀⾏」,并且「添加⼀列」后,所有的值都为 0 即可。


  • 4. 填表顺序:

根据「状态转移⽅程」,填表的顺序是「从上往下填写每⼀⾏」,「每⼀⾏从左往右」。


  • 5. 返回值:

根据「状态表⽰」,我们应该返回 dp[m][n] 的值。


代码如下:

class Solution {
public:int maxValue(vector<vector<int>>& grid) {int len = grid.size();int wide = grid[0].size();vector<vector<int>> dp(len + 1, vector<int>(wide + 1));for(int i = 1; i <= len; i++){for(int j = 1; j <= wide; j++){dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]) + grid[i - 1][j - 1];}}return dp[len][wide];}
};

性能分析:

  • 时间复杂度:O(mn)。

  • 空间复杂度:O(mn)。


以上便是本期动态规划的几道题目,大家做题时按照上述的“五步走”战略去分析思考,我相信大家都可以做对,


http://wed.xjx100/news/201141.html

相关文章

MySQL—日志

文章目录 一、错误日志二、二进制日志2.1 介绍2.2 格式2.3 查看2.4 删除 三、查询日志四、慢查询日志 一、错误日志 错误日志是 MySQL 中最重要的日志之一&#xff0c;它记录了当 mysqld 启动和停止时&#xff0c;以及服务器在运行过程中发生任何严重错误时的相关信息。当数据…

常用在线工具,非常实用,快收藏起来!

作者丨黑蛋 今天给大家介绍一些常用到的在线工具&#xff0c;能方便我们的日常学习&#xff1a; 编码工具&#xff1a; AES加密解密&#xff1a;http://www.jsons.cn/aesencrypt/ DNA编码解码&#xff1a;https://web.expasy.org/translate/ 双16进制编码解码&#xff1a;ht…

注册ChatGPT时提示Oops! The email you provided is not supported

问题描述 今天本想出一个ChatGPT的注册与使用的教程&#xff0c;结果上来吃了个闭门羹。之前我通过微软账号登录验证是没有问题的&#xff0c;但这次想使用另一个微软账号&#xff0c;结果提示Oops! The email you provided is not supported&#xff08;您提供的电子邮件不支…

操作系统基础知识介绍之系统安全术语解释

a Custom Performance Counter (PMC)&#xff0c;自定义性能计数器&#xff1a;这是一种用于测量程序执行时的各种性能指标的工具&#xff0c;如指令数、分支数、缓存命中率等。它通过在编译器或硬件中插入一些特殊的指令或寄存器来实现。它可以用于优化程序性能、分析程序行为…

使用VSCode创建第一个ESP-IDF项目

1.在VSCode中安装ESP-IDF: 在 VS Code 中安装 ESP-IDF&#xff1a; 在-VS-Code-中安装-ESP-IDF、新建项目 安装过程中可能会遇到的问题&#xff1a; 解决-pip-安装第三方包时因-SSL-报错_pip-ssl error 在完全使用VSCode安装ESP-IDF环境后&#xff0c;不会存在ESP-IDF Termin…

SQL语言分类

数据查询语言DQL 数据操纵语言DML 数据定义语言DDL 数据控制语言DCL 1&#xff0e;数据查询语言DQL 数据查询语言Dor基本结构是由SELECT子句&#xff0c;FROM子句&#xff0c;WHERE子句组成的查询块:SELECT FROM WHERE 2.数据操纵语言DML 数据操纵语言DM主要有三种形式…

YOLO V1-V3 简单介绍

目录 1. YOLO 2. YOLO V1 3. YOLO V2 4. YOLO V3 5. YOLO V3 SPP网络 5.1 Mosaic 图像增强 5.2 SPP 模块 5.3 CIou Loss 5.4 Focal loss 1. YOLO YOLO 是目标检测任务强大的算法&#xff0c;将目标检测的问题转换边界框和相关概率的回归问题&#xff0c;是目标检测…

【HTTPS加密】

&#x1f389;&#x1f389;&#x1f389;点进来你就是我的人了博主主页&#xff1a;&#x1f648;&#x1f648;&#x1f648;戳一戳,欢迎大佬指点! 欢迎志同道合的朋友一起加油喔&#x1f93a;&#x1f93a;&#x1f93a; 目录 1.HTTPS 是什么 1.1 运营商劫持 1.2 关于加密…