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;
}
}