首页 > 其他分享 >力扣刷题 45.跳跃游戏 II

力扣刷题 45.跳跃游戏 II

时间:2024-03-30 13:29:22浏览次数:21  
标签:力扣 元素 下标 nums int 45 II 跳跃 覆盖范围

目录

题干

解题思路


题干

给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]

每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说,如果你在 nums[i] 处,你可以跳转到任意 nums[i + j] 处:

  • 0 <= j <= nums[i] 
  • i + j < n

返回到达 nums[n - 1] 的最小跳跃次数。生成的测试用例可以到达 nums[n - 1]

示例 1:

输入: nums = [2,3,1,1,4]
输出: 2
解释: 跳到最后一个位置的最小跳跃数是 2。
     从下标为 0 跳到下标为 1 的位置,跳 1步,然后跳 3步到达数组的最后一个位置。

示例 2:

输入: nums = [2,3,0,1,4]
输出: 2

解题思路

贪心算法的核心思想是选择每一阶段的局部最优,从而达到全局最优。

因此,我们以数组遍历元素作为题目的大背景,每遍历一个元素,就是一个阶段,在这个阶段下去做操作,而这个操作要是当前阶段最优的。

for(int i = 0;i <nums.size();i++){

本题的目标是使用最少的跳跃步数达到数组的最后一个位置,那么怎么才能实现这一目标呢?

我们可能会想到每次跳当前元素的最大距离,但是这其实是错误的,让我们看一组例子,当前我们在下标为0,最大跳跃距离为2的格子,它的覆盖范围为从0到2,如果我们每次跳当前元素的最大距离,那么就会跳到下标为2,跳跃距离为1的元素,后面的过程就不一一罗列,整体的跳跃次数为3,下标由0到2到3再到4。

很明显这不是最优解,如果在下标为1的元素起跳,只需两步就到终点了。

最优策略应该是在当前覆盖范围内中,选出具有最大跳跃能力的元素,作为下一个起跳点。这样的意义是实现了每一阶段可以覆盖最大的范围,全局上就可以最优地覆盖到终点。而这也正是本题的难点,不太容易想到。

那么现在我们的目标是在当前覆盖范围内中,选出具有最大跳跃能力的元素。但是怎么把这一策略转换成代码?

作为下一个起跳点选出最大跳跃能力的元素很简单,可以用max函数更新。但是现在的问题是怎么用代码实现在覆盖范围内,用遍历?还是用index记录?还有下一个起跳点怎么实现,用回溯吗?

我们假设从第一个元素开始遍历,下标为0,它的最大跳跃距离是2,因此它可以实现的覆盖范围为从0到2。 当我们移动指针,指到下标为1的元素时,我们要记录他的值,以找到再覆盖范围内具有最大跳跃 能力的元素,  这里有个点需要注意,我们记录的值是它的跳跃距离,还是它所能更新的覆盖范围。这里 我们先记录它的最大跳跃距离。
for(int i = 0;i <nums.size();i++){
       next =  max(next,nums[i]);

然后我们移动指针指向下标为2的元素,记录他的值,这时我们已经遍历到覆盖范围的末端了,那么机器怎么知道已经到覆盖范围的末端了呢,我们很自然地联想到增加一个覆盖范围的变量cover,再使用一个判断语句去判断i是否等于覆盖范围cover。

}
    for(int i = 0;i <nums.size();i++){
       next =  max(next,nums[i]);
       if(i == cover){

下一步就是要从覆盖范围中,找到具有最大跳跃距离的元素。易知这个元素是下标为1的元素,我们要选这个元素为起跳点,但是我们的指针已经超过了这个元素,难不成要回溯回去吗?

其实不用,我们可以把覆盖范围cover更新,这样就在效果上实现了选下标为1的元素作为起跳点。因此,我们之前记录的值应该是覆盖范围cover,也就是nums[i]+i。这时,我们选定后下标为1的元素后,也就意味着实现了一次跳跃,步数要加一。修改和增加后的代码如下

   int cover = 0;
   int next = 0;
   int result = 0;
    
    for(int i = 0;i <nums.size();i++){
       next =  max(next,nums[i] + i);
       if(i == cover){
             cover = next;
             result ++;

下一步我们发现cover这个时候为4,已经覆盖到终点了,所以应该可以直接返回步数了。

所以我们在后面增加一个判断语句去判断覆盖范围是否大于等于数组末元素的下标,如果满足直接跳出循环。如果小于那就继续循环。

int jump(vector<int>& nums) {
   int cover = 0;
   int next = 0;
   int result = 0;

    for(int i = 0;i <nums.size();i++){
       next =  max(next,nums[i] + i);
       if(i == cover){
             cover = next;
             result ++;
             if(cover>=nums.size()-1){
                 break;
              }
       }
    }
    return result;
    }

在提交代码后,我们发现出错了,检查用例后,发现是当数组仅有一个元素的时候,不用跳跃就已经到终点,所以我们直接做一个剪枝操作,用一个if判断语句,跳出循环。

完整代码如下

class Solution {
public:
    int jump(vector<int>& nums) {
   int cover = 0;
   int next = 0;
   int result = 0;
    if(nums.size() == 1){
        return result;
    }
    for(int i = 0;i <nums.size();i++){
       next =  max(next,nums[i] + i);
       if(i == cover){
             cover = next;
             result ++;
             if(cover>=nums.size()-1){
                 break;
              }
       }
    }
    return result;
    }
};

标签:力扣,元素,下标,nums,int,45,II,跳跃,覆盖范围
From: https://blog.csdn.net/never1624/article/details/137170329

相关文章

  • LeetCode Python - 80. 删除有序数组中的重复项 II
    目录题目描述解法运行结果题目描述给你一个有序数组nums,请你原地删除重复出现的元素,使得出现次数超过两次的元素只出现两次,返回删除后数组的新长度。不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(1)额外空间的条件下完成。说明:为什么返回......
  • yii2 模型
    yii2模型Yii2的模型(Model)是MVC(Model-View-Controller)设计模式中的一部分,它代表业务数据、规则和逻辑的对象。模型通常用于处理与数据相关的业务逻辑,如数据的验证、访问和修改等。模型示例代码<?phpnamespaceapp\models;useYii;useyii\db\ActiveRecord;......
  • yii2 配置bootstrap使用
    yii2配置bootstrap使用配置config/web.php添加people<?php...$config=[...'bootstrap'=>['log','people'],...'components'=>['people'=>['class'......
  • yii2 Gii使用和自定义模板
    yii2Gii使用和自定义模板配置开启giiconfig/web.php添加代码if(YII_ENV_DEV){$config['bootstrap'][]='gii';$config['modules']['gii']=['class'=>'yii\gii\Module',];}入口脚本web......
  • yii2控制器
    yii2控制器Yii2的控制器(Controller)是MVC(Model-View-Controller)设计模式中的核心组件之一,负责处理用户请求并生成相应的响应。控制器包含了处理请求所需的方法(通常称为动作方法或动作),并可以调用模型和视图来执行相应的业务逻辑和展示内容。在Yii2中,控制器类通常继承自yii\we......
  • yii2安装
    yii2安装安装composercurl-sShttps://getcomposer.org/installer|phpmvcomposer.phar/usr/local/bin/composer安装yii2-basiccomposercreate-project--prefer-dist--stability=devyiisoft/yii2-app-basicyii2-basicnginx配置server{listen......
  • P4556 [Vani有约会] 雨天的尾巴 /【模板】线段树合并
    P4556[Vani有约会]雨天的尾巴/【模板】线段树合并在这题里面讲一下线段树合并。顾名思义就是把多个线段树合并成一个。显然完全二叉线段树(也就是普通线段树)是无法更高效的合并的,只能把所有节点加起来建个新树。但是在动态开点线段树中,有时候一个树只有几条链,这时候我们就是可......
  • 代码随想录第22天 | 235. 二叉搜索树的最近公共祖先 701.二叉搜索树中的插入操作 450.
    235. 二叉搜索树的最近公共祖先 235.二叉搜索树的最近公共祖先-力扣(LeetCode)代码随想录(programmercarl.com)二叉搜索树找祖先就有点不一样了!|235.二叉搜索树的最近公共祖先_哔哩哔哩_bilibili给定一个二叉搜索树,找到该树中两个指定节点的最近公共祖先。百度百......
  • 代码随想录训练营Day31:● 理论基础 ● 455.分发饼干 ● 376. 摆动序列 ● 53. 最大子
    理论基础贪心基础455.分发饼干题目链接https://leetcode.cn/problems/assign-cookies/description/题目描述思路自己写的,因为没有事先对两个数组进行排序,所以出现了问题classSolution{publicintfindContentChildren(int[]g,int[]s){Arrays.s......
  • 【力扣】300. 最长递增子序列(DFS+DP两种方法实现)
    目录题目传送最长递增子序列[DFS方法]DFS方法思路图思路简述代码大家可以自行考虑有没有优化的方法最长递增子序列[DP]方法DP方法思路图思路简述代码方案题目传送原题目链接最长递增子序列[DFS方法]DFS方法思路图思路简述对于序列中的每一个数字只有选择......