【LeetCode 热题100】45:跳跃游戏 II(详细解析)(Go语言版)

简介: 本文详细解析了力扣第45题“跳跃游戏II”的三种解法:贪心算法、动态规划和反向贪心。贪心算法通过选择每一步能跳到的最远位置,实现O(n)时间复杂度与O(1)空间复杂度,是面试首选;动态规划以自底向上的方式构建状态转移方程,适合初学者理解但效率较低;反向贪心从终点逆向寻找最优跳点,逻辑清晰但性能欠佳。文章对比了各方法的优劣,并提供了Go语言代码实现,助你掌握最小跳跃次数问题的核心技巧。

🚀 力扣 45:跳跃游戏 II(全解法详解)

📌 题目描述

给你一个非负整数数组 nums,表示你最初位于数组的第一个位置。

数组中的每个元素表示你在该位置可以跳跃的最大长度。

你的目标是使用 最少的跳跃次数 到达数组的最后一个位置。

假设你总是可以到达数组的最后一个位置。

🎯 示例 1:

输入: nums = [2,3,1,1,4]
输出: 2
解释: 跳到位置 1(跳 1 步),然后跳 4(再跳 1 步)——共 2 步

🎯 示例 2:

输入: nums = [2,3,0,1,4]
输出: 2

💡 解题思路总览

我们要找到 从起点跳到终点的最少步数。这题与「跳跃游戏 I」不同的地方是,这次我们不是判断“能否到达”,而是要求最小的“跳跃次数”。


✅ 解法一:贪心算法(最推荐🔥)

✨ 核心思路:

使用贪心思想,在每一步跳跃时,选择能跳的最远的那个位置。

🧠 实现步骤:

  1. 初始化步数 steps = 0,当前跳跃的边界 end = 0,和当前最远位置 farthest = 0
  2. 遍历数组:
    • 每次更新当前最远能跳到的位置
    • 当到达当前边界时,步数加一,并更新边界为 farthest

💻 代码实现(Go):

func jump(nums []int) int {
   
    steps := 0
    end := 0
    farthest := 0

    for i := 0; i < len(nums)-1; i++ {
   
        farthest = max(farthest, i+nums[i])
        if i == end {
   
            steps++
            end = farthest
        }
    }
    return steps
}

func max(a, b int) int {
   
    if a > b {
   
        return a
    }
    return b
}

⏱️ 时间复杂度:O(n)

🧠 空间复杂度:O(1)


🔄 解法二:动态规划(自底向上)

✨ 核心思路:

定义 dp[i] 表示从位置 i 跳到终点所需的最小步数。

🧠 实现步骤:

  1. 初始化 dp[n-1] = 0(终点到终点步数为0)
  2. 从右向左遍历数组
  3. 对于每个 i,从 i+1i+nums[i]dp[j] 中最小值

💻 代码实现(Go):

func jump(nums []int) int {
   
    n := len(nums)
    dp := make([]int, n)
    for i := n - 2; i >= 0; i-- {
   
        minStep := int(1e9)
        for j := i + 1; j <= i+nums[i] && j < n; j++ {
   
            if dp[j] < minStep {
   
                minStep = dp[j]
            }
        }
        dp[i] = minStep + 1
    }
    return dp[0]
}

⏱️ 时间复杂度:O(n²)

🧠 空间复杂度:O(n)

💬 优点:清晰易懂,适合入门
缺点:效率较低,不适合数据量大的情况


🔁 解法三:反向贪心(从终点往回跳)

✨ 核心思路:

从终点出发,每次选择最靠前、但能够跳到当前终点的位置,更新目标点,直到回到起点。

🧠 实现步骤:

  1. 初始位置设为 pos = n-1
  2. 遍历数组从头找哪个位置能跳到 pos
  3. 每次成功找到,就将 pos 更新为当前索引,同时步数加一

💻 代码实现(Go):

func jump(nums []int) int {
   
    pos := len(nums) - 1
    steps := 0
    for pos > 0 {
   
        for i := 0; i < pos; i++ {
   
            if i+nums[i] >= pos {
   
                pos = i
                steps++
                break
            }
        }
    }
    return steps
}

⏱️ 时间复杂度:O(n²)

🧠 空间复杂度:O(1)

💬 优点:逻辑反向,易于验证正确性
缺点:效率低,不适合大数据量


📊 三种方法对比总结

解法 时间复杂度 空间复杂度 适合场景 特点说明
✅ 贪心 O(n) O(1) 面试首选 高效实用,核心技巧
动态规划 O(n²) O(n) 教学/入门 思路清晰,但慢
反向贪心 O(n²) O(1) 验证/反向思考 逆向推理,有趣但低效

📌 最佳实践建议

  • 面试中建议优先写贪心算法,能体现算法思维和代码能力;
  • 若初学 DP,可尝试动态规划方式,加深状态转移理解;
  • 想换个角度思考,可尝试反向贪心,拓展思维维度。

📚 拓展阅读

  • Leetcode 55. 跳跃游戏 I
  • Leetcode 134. 加油站(贪心)
  • Leetcode 122. 买卖股票的最佳时机 II(贪心)

💡 如果你觉得这篇文章对你有帮助,欢迎点赞 👍、收藏 ⭐、关注 📌,我将持续更新更多 LeetCode 热题的高质量解析~


目录
相关文章
|
6天前
|
Go 开发者 索引
【LeetCode 热题100】路径与祖先:二叉树中的深度追踪技巧(力扣33 / 81/ 153/154)(Go语言版)
本文深入探讨了LeetCode中四道关于「搜索旋转排序数组」的经典题目,涵盖了无重复和有重复元素的情况。通过二分查找的变形应用,文章详细解析了每道题的解题思路和Go语言实现代码。关键点包括判断有序区间、处理重复元素以及如何缩小搜索范围。文章还总结了各题的异同,并推荐了类似题目,帮助读者全面掌握二分查找在旋转数组中的应用。无论是初学者还是有经验的开发者,都能从中获得实用的解题技巧和代码实现方法。
37 14
|
6天前
|
存储 算法 Go
【LeetCode 热题100】17:电话号码的字母组合(详细解析)(Go语言版)
LeetCode 17题解题思路采用回溯算法,通过递归构建所有可能的组合。关键点包括:每位数字对应多个字母,依次尝试;递归构建下一个字符;递归出口为组合长度等于输入数字长度。Go语言实现中,使用map存储数字到字母的映射,通过回溯函数递归生成组合。时间复杂度为O(3^n * 4^m),空间复杂度为O(n)。类似题目包括括号生成、组合、全排列等。掌握回溯法的核心思想,能够解决多种排列组合问题。
34 11
|
4天前
|
Go
【LeetCode 热题100】155:最小栈(详细解析)(Go语言版)
本文详细解析了力扣热题155:最小栈的解题思路与实现方法。题目要求设计一个支持 push、核心思路是使用辅助栈法,通过两个栈(主栈和辅助栈)来维护当前栈中的最小值。具体操作包括:push 时同步更新辅助栈,pop 时检查是否需要弹出辅助栈的栈顶,getMin 时直接返回辅助栈的栈顶。文章还提供了 Go 语言的实现代码,并对复杂度进行了分析。此外,还介绍了单栈 + 差值记录法的进阶思路,并总结了常见易错点,如 pop 操作时忘记同步弹出辅助栈等。
23 6
|
6天前
|
Go 索引
【LeetCode 热题100】739:每日温度(详细解析)(Go语言版)
这篇文章详细解析了 LeetCode 第 739 题“每日温度”,探讨了如何通过单调栈高效解决问题。题目要求根据每日温度数组,计算出等待更高温度的天数。文中推荐使用单调递减栈,时间复杂度为 O(n),优于暴力解法的 O(n²)。通过实例模拟和代码实现(如 Go 语言版本),清晰展示了栈的操作逻辑。此外,还提供了思维拓展及相关题目推荐,帮助深入理解单调栈的应用场景。
34 6
|
8月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
117 6
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
9月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
98 6
|
9月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
203 2
|
6月前
|
机器学习/深度学习 人工智能 自然语言处理
280页PDF,全方位评估OpenAI o1,Leetcode刷题准确率竟这么高
【10月更文挑战第24天】近年来,OpenAI的o1模型在大型语言模型(LLMs)中脱颖而出,展现出卓越的推理能力和知识整合能力。基于Transformer架构,o1模型采用了链式思维和强化学习等先进技术,显著提升了其在编程竞赛、医学影像报告生成、数学问题解决、自然语言推理和芯片设计等领域的表现。本文将全面评估o1模型的性能及其对AI研究和应用的潜在影响。
173 1
|
8月前
|
数据采集 负载均衡 安全
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
本文提供了多个多线程编程问题的解决方案,包括设计有限阻塞队列、多线程网页爬虫、红绿灯路口等,每个问题都给出了至少一种实现方法,涵盖了互斥锁、条件变量、信号量等线程同步机制的使用。
113 3
LeetCode刷题 多线程编程九则 | 1188. 设计有限阻塞队列 1242. 多线程网页爬虫 1279. 红绿灯路口
|
9月前
|
索引 Python
【Leetcode刷题Python】从列表list中创建一颗二叉树
本文介绍了如何使用Python递归函数从列表中创建二叉树,其中每个节点的左右子节点索引分别是当前节点索引的2倍加1和2倍加2。
151 7