题目所属分类
DP
原题链接
给你一个非负整数数组 nums ,你最初位于数组的第一个位置。
数组中的每个元素代表你在该位置可以跳跃的最大长度。
你的目标是使用最少的跳跃次数到达数组的最后一个位置。
假设你总是可以到达数组的最后一个位置。
代码案例:输入: nums = [2,3,1,1,4]
输出: 2
解释: 跳到最后一个位置的最小跳跃数是 2。
从下标为 0 跳到下标为 1 的位置,跳 1 步,然后跳 3 步到达数组的最后一个位置。
题解
通过枚举单纯的动态规划我们可以知道,f[N]数组里面是单调递增的
大概就是0,1,1,2,2,2,…
所以f[i]是一个单调递增的数组
又从动态规划的状态转移方程可知
f[i] = f[j] + 1
我们要枚举i之前的能跳到i的所有j,然后每次找到一个符合条件的j之后就f[i] = f[j]+1
因为初始的时候f[j]都为0,所以不管找到多少个j,都只会使得f[i]在0的基础上加1
找到第一个能跳到i的j的时候更新了一次f[i],之后无论再找到多少个j都只能使得f[i] = 0+1 = 1
也就是说除了第一个点之外,后面找到的点都是进行的重复的操作
所以我们只用找到第一个能跳到i的点j,然后用j去更新i的状态即f[i] = f[j] + 1
后面更新更多的点同理,只用找到能跳到的第一个点即可
动态规划时瓶颈就在于更新每个点的最小值时需要遍历所有能跳到i的点,而有了单调性以后就可以用第一个能跳到i的点更新了
因为找到第一个点和遍历所有的点都只遍历了一次,所以时间复杂度会降到O(n)
class Solution {
public int jump(int[] nums) {
int n = nums.length;
int[] f = new int[n + 10];
for(int i = 1 , j = 0 ; i < n ; i++){
while(j + nums[j] < i ) j++;
f[i] = f[j]+1 ;
}
return f[n-1];
}
}