力扣日记3.19-【贪心算法篇】55. 跳跃游戏

力扣日记:【贪心算法篇】55. 跳跃游戏

日期:2024.3.19

参考:代码随想录、力扣

55. 跳跃游戏

题目描述

难度:中等

给你一个非负整数数组 nums ,你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。

判断你是否能够到达最后一个下标,如果可以,返回 true ;否则,返回 false 。

示例 1:

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

输出:true

解释:可以先跳 1 步,从下标 0 到达下标 1, 然后再从下标 1 跳 3 步到达最后一个下标。

示例 2:

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

输出:false

解释:无论怎样,总会到达下标为 3 的位置。但该下标的最大跳跃长度是 0 , 所以永远不可能到达最后一个下标。

提示:

  • 1 <= nums.length <= 10^4
  • 0 <= nums[i] <= 10^5

题解

cpp 复制代码
class Solution {
public:
#define SOLUTION 2
#if SOLUTION == 1   /*回溯法,超出时间限制*/
    bool canJump(vector<int>& nums) {
        return backtracking(nums, 0);
    }
    bool backtracking(vector<int>& nums, int startindex) {
        // 终止条件:如果已经到达最后一个下标
        if (startindex >= nums.size() - 1) {
            return true;
        }
        // for 循环
        // 从大到小横向遍历这个元素值(选择可以跳跃的长度)
        for (int i = nums[startindex]; i > 0; i--) {
            startindex += i;    // 跳到下一个位置
            if (backtracking(nums, startindex)) return true;
            // 如果当前跳跃步数无法到最后一个坐标,则回溯
            startindex -= i;
        }
        return false;
    }
#elif SOLUTION == 2
    bool canJump(vector<int>& nums) {
        // 看最大覆盖范围!但每次只移动一个长度(关键!!!),并不断更新最大覆盖范围(最大可到达的下标)
        // 局部最优:获取每个位置当前可到达的最大下标
        // 全局最优:遍历可到达的全部元素,找到其中最大的可到达下标即为总的最大下标
        int curMaxIndex = 0;
        for (int i = 0; i <= curMaxIndex; i++) {    // 最大只能到curMaxIndex(不断更新)
            if (i + nums[i] > curMaxIndex) {    // 如果新的当前位置可到达的最大长度 比 curMaxIndex 大,则更新
                curMaxIndex = i + nums[i];
            }
            if (curMaxIndex >= nums.size() - 1) {   // 如果已能到达最后一个下标,则返回true
                return true;
            }
        }
        return false;
    }
#endif
};

复杂度

贪心算法:

  • 时间复杂度:O(n)
  • 空间复杂度:O(1)

思路总结

  • 一开始尝试用回溯算法,虽然思路是正确的,但回溯法相当于穷举,结果超出时间范围了
  • 参考代码随想录的思路:
    • 看最大覆盖范围!注意每次只移动一个长度(关键!!!),并不断更新最大覆盖范围(即最大可到达的下标)

      • 这样的话,就可以不管究竟在每个位置要跳几步,只要知道在覆盖范围内的每个位置能到达的最大下标即可(能覆盖到最大下标即成功)。
    • 局部最优:获取每个位置当前可到达的最大下标

    • 全局最优:遍历覆盖范围内可到达全部元素,找到其中最大的可到达下标即为总的最大下标

相关推荐
DdddJMs__13519 分钟前
C语言 | Leetcode C语言题解之第557题反转字符串中的单词III
c语言·leetcode·题解
Sunyanhui11 小时前
力扣 二叉树的直径-543
算法·leetcode·职场和发展
一个不喜欢and不会代码的码农1 小时前
力扣105:从先序和中序序列构造二叉树
数据结构·算法·leetcode
AnFany7 小时前
LeetCode【0051】N皇后
python·算法·leetcode·回溯法·n皇后
可别是个可爱鬼7 小时前
代码随想录 -- 动态规划 -- 完全平方数
数据结构·python·算法·leetcode·动态规划
一直学习永不止步7 小时前
LeetCode题练习与总结:至少有 K 个重复字符的最长子串--395
java·算法·leetcode·字符串·滑动窗口·哈希表·分治
weixin_478689768 小时前
贪心算法理论
算法·贪心算法
DdddJMs__1359 小时前
C语言 | Leetcode C语言题解之第552题学生出勤记录II
c语言·leetcode·题解
DdddJMs__1359 小时前
C语言 | Leetcode C语言题解之第554题砖墙
c语言·leetcode·题解
weixin_478689769 小时前
【121. 买卖股票的最佳时机】——贪心算法/动态规划
算法·贪心算法·动态规划