LeetCode 45. 跳跃游戏 II【贪心算法】

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

提示:

1 <= nums.length <= 10^4
0 <= nums[i] <= 1000
题目保证可以到达 nums[n-1]

  • 贪心的思路,局部最优:当前可移动距离尽可能多走,如果还没到终点,步数再加一。整体最优:一步尽可能多走,从而达到最小步数。
  • 思路虽然是这样,但在写代码的时候还不能真的能跳多远就跳多远,那样就不知道下一步最远能跳到哪里了。
  • 所以真正解题的时候,要从覆盖范围出发,不管怎么跳,覆盖范围内一定是可以跳到的,以最小的步数增加覆盖范围,覆盖范围一旦覆盖了终点,得到的就是最小步数!
  • 这里需要统计两个覆盖范围,当前这一步的最大覆盖和下一步最大覆盖
  • 如果移动下标达到了当前这一步的最大覆盖最远距离了,还没有到终点的话,那么就必须再走一步来增加覆盖范围,直到覆盖范围覆盖了终点。
class Solution {
    public int jump(int[] nums) {
        if (nums.length == 1) return 0; // 起点就是终点
        int cur = 0;    // 当前覆盖最远距离下标
        int res = 0;    // 记录走的最大步数
        int next = 0;   // 下一步覆盖最远距离下标
        for (int i = 0; i < nums.length; i++) {
            next = Math.max(nums[i] + i, next); // 更新下一步覆盖最远距离下标
            if (i == cur) { // 如果到了当前覆盖最远距离下标
                if (cur < nums.length - 1) {    // 如果当前覆盖最远距离下标不是终点
                    res++;  // 需要走下一步
                    cur = next; // 更新当前覆盖最远距离下标(相当于加油了)
                    if (next >= nums.length - 1) break; // 下一步的覆盖范围已经可以达到终点,结束循环
                }else break;    // 当前覆盖最远距到达集合终点,不用做res++操作了,直接结束
            }
        }
        return res;
    }
}
class Solution {
    public int jump(int[] nums) {
        if (nums.length == 1) return 0;
        int count = 0;  // 记录跳跃的次数
        int curDistance = 0;    // 当前的覆盖最大区域
        int maxDistance = 0;    // 最大的覆盖区域
        for (int i = 0; i < nums.length; i++) {
            maxDistance = Math.max(maxDistance, i + nums[i]);   // 在可覆盖区域内更新最大的覆盖区域
            if (maxDistance >= nums.length - 1) {   // 说明当前一步,再跳一步就到达了末尾
                count++;
                break;
            }
            if (i == curDistance) { // 走到当前覆盖的最大区域时,更新下一步可达的最大区域
                curDistance = maxDistance;
                count++;
            }
        }
        return count;
    }
}
  • 简化版
class Solution {
    public int jump(int[] nums) {
        int res = 0;
        int curDistance = 0;    // 当前覆盖的最远距离下标
        int nextDistance = 0;   // 下一步覆盖的最远距离下标
        for (int i = 0; curDistance < nums.length - 1; i++) {
            nextDistance = Math.max(nextDistance, i + nums[i]); // 更新下一步覆盖的最远距离下标
            // 可达位置的改变次数就是跳跃次数
            if (i == curDistance) { // 遇到当前覆盖的最远距离下标
                curDistance = nextDistance; // 更新当前覆盖的最远距离下标
                res++;
            }
        }
        return res;
    }
}
  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值