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];
    }
}
posted @   依嘫  阅读(42)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)
点击右上角即可分享
微信分享提示