78、贪心-跳跃游戏

思路

方法1: canJump01 - 使用递归(回溯法)

这个方法是通过递归实现的,它从数组的第一个位置开始,尝试所有可能的跳跃步数,直到达到数组的最后一个位置或遍历完所有的可能性。

思路:

  • 如果数组为空或者长度为0,直接返回 false
  • 从数组的第一个位置(index = 0)开始调用递归函数 process
  • 递归的终止条件是,如果当前的 index 等于 N-1(数组的最后一个位置),则返回 true
  • 在当前位置,根据该位置的数字决定可以跳跃的步数范围,递归地尝试每一种跳跃步数。
  • 如果任何一种跳跃方式可以到达最后位置,则返回 true

缺点:

  • 这种方法的时间复杂度非常高,因为它尝试了所有可能的路径,可能导致指数级的计算量。

方法2: canJump02 - 使用动态规划

这个方法使用了动态规划(DP)来减少重复计算,提高效率。

思路:

  • 初始化一个布尔型的数组 dp,其中 dp[i] 表示是否可以从起始位置跳跃到位置 i
  • dp[0] 初始化为 true,因为起始位置总是可达的。
  • i = 1 开始遍历数组,对于每个位置 i,检查所有之前的位置 j,看是否存在一个 j,使得从 j 跳跃到 i 是可行的(即 dp[j]truej + nums[j] 大于等于 i)。
  • 如果找到这样的 j,则设置 dp[i] = true 并中断当前循环。

优点:

  • 时间复杂度为 O(n^2),空间复杂度为 O(n)。

方法3: canJump - 贪心算法

使用贪心算法解决问题,思路更为直接和高效。

思路:

  • 维护一个变量 maxReach 来存储从起点开始可达的最远位置。
  • 遍历数组,对于每个位置 i,首先检查 i 是否超过了之前的 maxReach。如果超过,则说明无法到达当前位置,返回 false
  • 然后更新 maxReachmax(maxReach, i + nums[i])
  • 如果在任何时刻 maxReach 大于等于数组的最后位置,直接返回 true

优点:

  • 时间复杂度为 O(n),空间复杂度为 O(1),是三种方法中最优的。

代码如下:

java 复制代码
class Solution {
   public boolean canJump01(int[] nums) {
        if (nums==null||nums.length==0){
            return false;
        }
        return process(nums,0,nums.length);
    }

    private boolean process(int[] nums, int index, int N) {
        if (index==N-1){
            return true;
        }
        int num=nums[index];
        boolean ans=false;
        //当前节点跳 1-num 步
        for (int i = 1; i <=num; i++) {
            ans=ans||process(nums,index+i,N);
        }
        return ans;
    }


    public boolean canJump02(int[] nums){
        if (nums==null||nums.length==0){
            return false;
        }
        int N=nums.length;
        boolean[] dp = new boolean[N];
        dp[0] = true; // 起点是可达的
        for (int i = 1; i < nums.length; i++) {
            // 初始化当前位置为不可达
            dp[i] = false;
            // 遍历到当前位置之前的所有位置
            for (int j = 0; j < i; j++) {
                // 如果位置 j 可达,并且从 j 跳跃足够远以到达 i
                if (dp[j] && j + nums[j] >= i) {
                    dp[i] = true;
                    break; // 找到一个可达的位置后,无需继续检查
                }
            }
        }
        return dp[N-1];
    }

     //假设从 0 跳 最远跳到i位置 那说明 0-i都是可达的,所以只需要关心每次跳多远即可
    //然后再次从1 位置尝试跳最远  如果1 >max 说明在0位置跳的时候无法跳到1
    public boolean canJump(int[] nums) {
        int maxReach = 0; // 可达的最远位置
        for (int i = 0; i < nums.length; i++) {
            // 如果当前位置超过了最远可达位置,则无法继续
            if (i > maxReach) {
                return false;
            }
            // 更新最远可达位置
            maxReach = Math.max(maxReach, i + nums[i]);
            // 如果最远可达位置已经超过或到达数组的最后一个位置
            if (maxReach >= nums.length - 1) {
                return true;
            }
        }
        // 结束循环后,如果还没返回true,则说明最后位置不可达
        return false;
    }
}
相关推荐
WolfGang0073219 分钟前
代码随想录算法训练营 Day32 | 动态规划 part05
算法·动态规划
lulu121654407830 分钟前
Claude Code Harness架构技术深度解析:生产级AI Agent工程化实践
java·人工智能·python·ai编程
阿里加多32 分钟前
第 1 章:Go 并发编程概述
java·开发语言·数据库·spring·golang
碧海银沙音频科技研究院34 分钟前
1-1杰理蓝牙SOC的UI配置开发方法
人工智能·深度学习·算法
一 乐37 分钟前
物流信息管理|基于springboot + vue物流信息管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·物流信息管理系统
2301_792674861 小时前
java学习day29(juc)
java·开发语言·学习
啊我不会诶1 小时前
2024CCPC长春邀请赛
算法
珂朵莉MM1 小时前
第七届全球校园人工智能算法精英大赛-算法巅峰赛产业命题赛第3赛季优化题--启发式算法+操作因子设计
人工智能·算法
魔士于安2 小时前
Unity资源Toon City Pack 发电厂 工地 公园 地铁站口 银行 车 直升飞机 可动 URP
游戏·unity·游戏引擎·贴图·模型
希望永不加班2 小时前
SpringBoot 自动配置类加载顺序与优先级
java·spring boot·后端·spring·mybatis