LeetCode题目55:跳跃游戏【python5种算法贪心/回溯/动态规划/优化贪心/索引哈希映射 详解】

本文主要是介绍LeetCode题目55:跳跃游戏【python5种算法贪心/回溯/动态规划/优化贪心/索引哈希映射 详解】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目描述

给定一个非负整数数组 nums,你最初位于数组的第一个位置。数组中的每个元素代表你在该位置可以跳跃的最大长度。判断你是否能够到达最后一个位置。

输入格式
  • nums:一个由非负整数构成的数组。
输出格式
  • 返回一个布尔值,表示是否能到达最后一个位置。

示例

示例 1
输入: nums = [2,3,1,1,4]
输出: true
解释: 可以先跳 1 步,从位置 0 到 1,然后从位置 1 跳 3 步到最后一个位置。
示例 2
输入: nums = [3,2,1,0,4]
输出: false
解释: 无论如何,你总会到达位置 3,但该位置的最大跳跃长度是 0,所以你永远不可能到达最后一个位置。

方法一:贪心算法

解题步骤
  1. 初始化最远距离max_reach 初始化为 0,用于记录能到达的最远位置。
  2. 遍历数组:遍历数组中的每个位置,并更新 max_reach
  3. 判断可达性:如果在某个位置 imax_reach < i,说明无法继续向前跳跃,返回 false;如果 max_reach 大于等于数组的最后一个位置,则返回 true
完整的规范代码
def canJump(nums):"""使用贪心算法判断能否跳到最后一个位置:param nums: List[int], 输入的数组:return: bool, 是否能到达最后一个位置"""max_reach, n = 0, len(nums)for i in range(n):if i > max_reach:return Falsemax_reach = max(max_reach, i + nums[i])if max_reach >= n - 1:return Truereturn False# 示例调用
print(canJump([2,3,1,1,4]))  # 输出: True
print(canJump([3,2,1,0,4]))  # 输出: False
算法分析
  • 时间复杂度:(O(n)),其中 n 是数组 nums 的长度。
  • 空间复杂度:(O(1)),使用了常数额外空间。

方法二:回溯算法

解题步骤
  1. 定义回溯函数:尝试从当前位置开始,模拟所有可能的跳跃步数,向前递归探索。
  2. 递归跳跃:从当前位置起跳,逐步减小跳跃距离直到 0,递归调用自身。
  3. 终止条件:如果到达数组末尾,返回 true;如果所有跳跃尝试都失败,返回 false
完整的规范代码
def canJumpFromPosition(position, nums):if position == len(nums) - 1:return Truefurthest_jump = min(position + nums[position], len(nums) - 1)for next_position in range(position + 1, furthest_jump + 1):if canJumpFromPosition(next_position, nums):return Truereturn Falsedef canJump(nums):"""使用回溯算法判断能否跳到最后一个位置:param nums: List[int], 输入的数组:return: bool, 是否能到达最后一个位置"""return canJumpFromPosition(0, nums)# 示例调用
print(canJump([2,3,1,1,4]))  # 输出: True
print(canJump([3,2,1,0,4]))  # 输出: False
算法分析
  • 时间复杂度:(O(2^n)),其中 n 是数组 nums 的长度,每个位置都可能产生递归调用。
  • 空间复杂度:(O(n)),递归的深度最多为 n

方法三:动态规划(自底向上)

解题步骤
  1. 初始化状态数组dp[i] 表示从位置 i 开始是否能到达数组的末尾。
  2. 状态转移:从后向前填充 dp 数组,根据 dp[j] (对于所有 i < j <= i+nums[i]) 更新 dp[i]
  3. 结果返回:返回 dp[0]
完整的规范代码
def canJump(nums):"""使用动态规划算法判断能否跳到最后一个位置:param nums: List[int], 输入的数组:return: bool, 是否能到达最后一个位置"""n = len(nums)dp = [False] * ndp[n - 1] = Truefor i in range(n - 2, -1, -1):furthest_jump = min(i + nums[i], n - 1)for j in range(i + 1, furthest_jump + 1):if dp[j]:dp[i] = Truebreakreturn dp[0]# 示例调用
print(canJump([2,3,1,1,4]))  # 输出: True
print(canJump([3,2,1,0,4]))  # 输出: False
算法分析
  • 时间复杂度:(O(n^2)),其中 n 是数组 nums 的长度,对于每个元素,可能需要遍历 nums[i] 次。
  • 空间复杂度:(O(n)),存储状态数组 dp 需要 n 个额外空间。

方法四:优化的贪心算法

解题步骤
  1. 从后向前遍历:从数组的倒数第二个元素开始,向前遍历数组。
  2. 更新目标位置:如果当前位置加上可跳跃的最大长度大于或等于目标位置,则更新目标位置为当前位置。
  3. 结果判断:遍历结束后,如果目标位置更新为数组的起始位置,则返回 true,否则返回 false
完整的规范代码
def canJump(nums):"""使用优化的贪心算法判断能否跳到最后一个位置:param nums: List[int], 输入的数组:return: bool, 是否能到达最后一个位置"""n = len(nums)lastPos = n - 1  # 最初目标位置为数组的最后一个位置for i in range(n - 2, -1, -1):  # 从倒数第二个位置向前遍历if i + nums[i] >= lastPos:  # 如果当前位置能跳到或跳过目标位置lastPos = i  # 更新目标位置为当前位置return lastPos == 0  # 最终目标位置是否为起始位置# 示例调用
print(canJump([2,3,1,1,4]))  # 输出: True
print(canJump([3,2,1,0,4]))  # 输出: False
算法分析
  • 时间复杂度:(O(n)),其中 n 是数组 nums 的长度,只需遍历一次数组。
  • 空间复杂度:(O(1)),使用了常数的额外空间。

方法五:索引哈希映射

解题步骤
  1. 建立索引哈希映射:创建一个字典来记录每个元素能达到的最远距离。
  2. 判断可达性:从第一个元素开始,逐个检查每个元素是否可达,同时更新可达的最远距离。
  3. 递归结束条件:如果在某个点发现最远距离无法继续向前或已覆盖数组末尾,则停止。
完整的规范代码
def canJump(nums):"""使用索引哈希映射法判断能否跳到最后一个位置:param nums: List[int], 输入的数组:return: bool, 是否能到达最后一个位置"""max_reach = 0  # 初始化最远可达位置for i in range(len(nums)):if i > max_reach:return False  # 当前索引不可达max_reach = max(max_reach, i + nums[i])  # 更新最远可达位置if max_reach >= len(nums) - 1:return True  # 已可达数组末尾return False# 示例调用
print(canJump([2,3,1,1,4]))  # 输出: True
print(canJump([3,2,1,0,4]))  # 输出: False
算法分析
  • 时间复杂度:(O(n)),其中 n 是数组 nums 的长度,只需遍历一次数组。
  • 空间复杂度:(O(1)),使用了常数的额外空间。

不同算法的优劣势对比

特征方法一: 贪心算法方法二: 回溯算法方法三: 动态规划方法四: 优化贪心方法五: 索引哈希映射
时间复杂度(O(n))(O(2^n))(O(n^2))(O(n))(O(n))
空间复杂度(O(1))(O(n))(O(n))(O(1))(O(1))
优势快速简单理论上完备结构清晰更直观高效直观简洁
劣势基本无极度低效空间复杂需逆向思维与方法一类似

应用示例详解

跳跃游戏的算法在多个领域内都有广泛的应用,特别是在游戏开发、动画制作、机器学习等领域。以下是两个具体的应用场景,展示如何将跳跃游戏算法转化为实际可用的技术解决方案。

应用一:游戏开发中的关卡可玩性验证

在平台跳跃游戏的开发过程中,设计师常常需要创建出既具有挑战性又能确保玩家能够完成的关卡。使用跳跃游戏算法可以帮助开发者验证任何给定关卡的可玩性。

场景描述

  • 设计师创建了一个关卡,其中包含不同高度和间隔的平台,玩家从起点开始,需要通过跳跃到达终点。
  • 每个平台的数字表示玩家从该点最多可以向前跳跃的距离(即 nums 数组)。

技术实现

  • 开发者使用贪心算法,从起点开始,实时计算玩家可以到达的最远距离。
  • 如果在任一点,计算得到的最远距离小于该点的索引,意味着玩家无法从当前平台跳到下一个平台,关卡不可通行。

代码示例

def verify_level(nums):max_reach = 0for i, jump in enumerate(nums):if i > max_reach:return False, i  # 返回不可通行的最远点max_reach = max(max_reach, i + jump)if max_reach >= len(nums) - 1:return True, None  # 关卡可通行return False, len(nums) - 1# 关卡设计示例:玩家可以从每个位置跳跃的最大长度
level_design = [2, 3, 1, 0, 4, 2, 1]
is_playable, fail_point = verify_level(level_design)
print(f"Level Playable: {is_playable}, Failure Point: {fail_point}")

输出解析

  • 如果关卡可通行,函数返回 TrueNone
  • 如果关卡不可通行,函数指出玩家被卡住的具体位置。
应用二:动画制作中的运动路径规划

动画制作中,制定角色或物体的移动路径是一个核心任务,跳跃游戏算法可以用来确保动画中的运动路径是连贯和逻辑上可行的。

场景描述

  • 动画师需要创建一个场景,其中一个角色需要从场景的一端跳跃到另一端。
  • 每个可能的着陆点的数字表示从该点角色能够跳跃的最大长度。

技术实现

  • 使用贪心算法来确定角色的每一步是否都能顺利落到一个合法的着陆点。
  • 这样的算法保证了动画的连贯性和视觉上的合理性。

代码示例

def plan_animation_path(spots):reach = 0for i in range(len(spots)):if i > reach:return False  # 角色无法继续前进reach = max(reach, i + spots[i])return True  # 角色可以顺利完成动画# 动画路径设计示例:角色可以从每个位置跳跃的最大长度
animation_path = [1, 2, 0, 3, 2, 0, 1]
can_complete = plan_animation_path(animation_path)
print(f"Animation Path Complete: {can_complete}")

输出解析

  • 返回 True 表示角色可以顺利从起点跳到终点。
  • 返回 False 表示存在至少一个点使得角色无法从该点继续前进。

总结

通过上述应用示例,我们可以看到跳跃游戏算法不仅仅局限于理论问题,它可以广泛应用于实际项目中,如游戏开发和动画制作,帮助开发者和设计师验证和规划合理的路径和策略。这种算法的实际应用强调了线性代数在现代编程中的实用价值和广泛适用性。

这篇关于LeetCode题目55:跳跃游戏【python5种算法贪心/回溯/动态规划/优化贪心/索引哈希映射 详解】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/925485

相关文章

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

HDFS—存储优化(纠删码)

纠删码原理 HDFS 默认情况下,一个文件有3个副本,这样提高了数据的可靠性,但也带来了2倍的冗余开销。 Hadoop3.x 引入了纠删码,采用计算的方式,可以节省约50%左右的存储空间。 此种方式节约了空间,但是会增加 cpu 的计算。 纠删码策略是给具体一个路径设置。所有往此路径下存储的文件,都会执行此策略。 默认只开启对 RS-6-3-1024k

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个