leetcode 45. 跳跃游戏 II -java版本
题目所属分类
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];
}
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)