13-跳跃游戏 II

给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]

每个元素 nums[i] 表示从索引 i 向后跳转的最大长度。换句话说,如果你在 nums[i] 处,你可以跳转到任意 nums[i + j] 处:

  • 0 <= j <= nums[i] 
  • i + j < n

返回到达 nums[n - 1] 的最小跳跃次数。生成的测试用例可以到达 nums[n - 1]

方法一: 贪心算法

这是一个典型的贪心算法问题,要求计算到达数组最后一个位置的最小跳跃次数。以下是使用 TypeScript 实现的代码:

代码实现

function jump(nums: number[]): number {const n = nums.length;if (n === 1) {return 0;}// 当前跳跃能到达的最远位置let currentMaxReach = 0;// 下一步跳跃能到达的最远位置let nextMaxReach = 0;// 跳跃次数let jumps = 0;for (let i = 0; i < n - 1; i++) {// 更新下一步跳跃能到达的最远位置nextMaxReach = Math.max(nextMaxReach, i + nums[i]);// 如果当前位置到达了当前跳跃能到达的最远位置if (i === currentMaxReach) {// 进行一次跳跃jumps++;// 更新当前跳跃能到达的最远位置为下一步跳跃能到达的最远位置currentMaxReach = nextMaxReach;// 如果当前跳跃能到达的最远位置已经可以到达数组的最后一个位置,直接返回跳跃次数if (currentMaxReach >= n - 1) {break;}}}return jumps;
}// 示例调用
const nums = [2, 3, 1, 1, 4];
const minJumps = jump(nums);
console.log("到达数组最后一个位置的最小跳跃次数:", minJumps);

代码解释

  1. 边界条件处理:如果数组长度为 1,说明已经在最后一个位置,不需要跳跃,直接返回 0。
  2. 初始化变量
    • currentMaxReach:表示当前跳跃能到达的最远位置,初始化为 0。
    • nextMaxReach:表示下一步跳跃能到达的最远位置,初始化为 0。
    • jumps:表示跳跃次数,初始化为 0。
  3. 遍历数组
    • 在遍历过程中,不断更新 nextMaxReach,它是当前位置 i 加上该位置能跳跃的最大长度 nums[i] 与之前的 nextMaxReach 中的较大值。
    • 当 i 等于 currentMaxReach 时,说明已经到达了当前跳跃能到达的最远位置,此时需要进行一次跳跃,jumps 加 1。
    • 同时更新 currentMaxReach 为 nextMaxReach,表示进入下一次跳跃的范围。
    • 如果 currentMaxReach 已经大于等于数组的最后一个位置的索引 n - 1,说明已经可以到达最后一个位置,直接跳出循环。
  4. 返回结果:遍历结束后,jumps 即为到达数组最后一个位置的最小跳跃次数,将其返回。

复杂度分析

  • 时间复杂度:O(n),其中  是数组的长度。因为只需要对数组进行一次遍历。
  • 空间复杂度:O(1),只使用了常数级的额外变量。

这种贪心算法的核心思想是在每一次跳跃中,尽可能地跳到最远的位置,从而用最少的跳跃次数到达数组的最后一个位置。

方法二: 广度优先搜索

除了前面介绍的贪心算法,还可以使用广度优先搜索(BFS)的方法来解决这个问题。BFS 的核心思想是从起始位置开始,逐层扩展所有可能的跳跃位置,直到到达目标位置,每扩展一层就相当于进行了一次跳跃,记录扩展的层数即为最小跳跃次数。

代码实现

function jump(nums: number[]): number {const n = nums.length;if (n === 1) {return 0;}// 记录已经访问过的位置const visited = new Array(n).fill(false);// 初始化队列,队列中存储 [当前位置, 跳跃次数]const queue: [number, number][] = [[0, 0]];visited[0] = true;while (queue.length > 0) {const [currentIndex, jumps] = queue.shift()!;// 遍历当前位置能跳跃到的所有位置for (let j = 1; j <= nums[currentIndex] && currentIndex + j < n; j++) {const nextIndex = currentIndex + j;// 如果到达了最后一个位置,返回当前跳跃次数加 1if (nextIndex === n - 1) {return jumps + 1;}// 如果该位置未被访问过if (!visited[nextIndex]) {// 标记为已访问visited[nextIndex] = true;// 将该位置和跳跃次数加 1 加入队列queue.push([nextIndex, jumps + 1]);}}}return -1; // 理论上不会执行到这里,因为题目保证可以到达最后一个位置
}// 示例调用
const nums = [2, 3, 1, 1, 4];
const minJumps = jump(nums);
console.log("到达数组最后一个位置的最小跳跃次数:", minJumps);

代码解释

  1. 边界条件处理:如果数组长度为 1,说明已经在最后一个位置,不需要跳跃,直接返回 0。
  2. 初始化变量
    • visited:一个布尔类型的数组,用于记录每个位置是否已经被访问过,初始时所有位置都标记为未访问。
    • queue:一个队列,用于存储待扩展的位置和对应的跳跃次数,初始时将起始位置 0 和跳跃次数 0 加入队列,并将起始位置标记为已访问。
  3. BFS 过程
    • 当队列不为空时,从队列中取出一个元素,包含当前位置 currentIndex 和跳跃次数 jumps
    • 遍历当前位置能跳跃到的所有位置 nextIndex(范围是从 currentIndex + 1 到 currentIndex + nums[currentIndex] 且不超过数组长度)。
    • 如果 nextIndex 是最后一个位置,说明已经到达目标,返回当前跳跃次数加 1。
    • 如果 nextIndex 未被访问过,将其标记为已访问,并将 [nextIndex, jumps + 1] 加入队列。
  4. 返回结果:如果一切正常,在 BFS 过程中会找到到达最后一个位置的最小跳跃次数并返回;理论上不会执行到返回 -1 的情况,因为题目保证可以到达最后一个位置。

复杂度分析

  • 时间复杂度:O(n),其中  是数组的长度。每个位置最多被访问一次,因此时间复杂度为线性。
  • 空间复杂度:O(n),主要用于存储 visited 数组和队列,队列在最坏情况下可能存储所有位置。

与贪心算法相比,BFS 方法的代码实现相对复杂一些,但它的思路更加直观,适合用于解决一些需要搜索所有可能路径的问题。不过,在这个特定问题中,贪心算法的时间和空间复杂度虽然与 BFS 相同,但在实际运行中可能会更快,因为贪心算法不需要维护队列和访问标记数组。

方法三: 动态规划

动态规划的核心思想是将原问题分解为子问题,通过求解子问题的最优解来得到原问题的最优解。

思路分析

  1. 定义状态:设 dp[i] 表示到达数组中第 i 个位置所需的最小跳跃次数。
  2. 初始化状态dp[0] = 0,因为初始位置就在 nums[0],不需要跳跃。对于其他位置 i,初始化为一个较大的值(比如 Infinity),表示暂时无法到达。
  3. 状态转移方程:对于每个位置 i,遍历其前面的所有位置 j0 <= j < i),如果从位置 j 能够跳到位置 i(即 j + nums[j] >= i),则更新 dp[i] 为 dp[j] + 1 和 dp[i] 中的较小值。
  4. 最终结果dp[n - 1] 即为到达数组最后一个位置所需的最小跳跃次数。

代码实现

function jump(nums: number[]): number {const n = nums.length;// 初始化 dp 数组,dp[i] 表示到达第 i 个位置所需的最小跳跃次数const dp: number[] = new Array(n).fill(Infinity);// 初始位置不需要跳跃dp[0] = 0;// 遍历数组中的每个位置for (let i = 1; i < n; i++) {// 遍历 i 之前的所有位置for (let j = 0; j < i; j++) {// 如果从位置 j 能够跳到位置 iif (j + nums[j] >= i) {// 更新 dp[i] 为 dp[j] + 1 和 dp[i] 中的较小值dp[i] = Math.min(dp[i], dp[j] + 1);}}}return dp[n - 1];
}// 示例调用
const nums = [2, 3, 1, 1, 4];
const minJumps = jump(nums);
console.log("到达数组最后一个位置的最小跳跃次数:", minJumps);

复杂度分析

  • 时间复杂度:O(n^2),其中  是数组的长度。因为需要使用两层嵌套循环来填充 dp 数组。
  • 空间复杂度:O(n),主要用于存储 dp 数组。

代码解释

  1. 初始化 dp 数组:创建一个长度为 n 的数组 dp,并将所有元素初始化为 Infinity,表示暂时无法到达。将 dp[0] 设为 0,因为初始位置不需要跳跃。
  2. 填充 dp 数组:使用两层嵌套循环,外层循环遍历从 1 到 n - 1 的每个位置 i,内层循环遍历从 0 到 i - 1 的每个位置 j。对于每个位置 j,如果从位置 j 能够跳到位置 i(即 j + nums[j] >= i),则更新 dp[i] 为 dp[j] + 1 和 dp[i] 中的较小值。
  3. 返回结果:最终返回 dp[n - 1],即到达数组最后一个位置所需的最小跳跃次数。

动态规划方法虽然可以解决这个问题,但时间复杂度较高,在处理大规模数组时性能可能不如贪心算法。贪心算法的时间复杂度为 ,是更优的解决方案。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/70152.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Qt的QToolBox的使用

QToolBox 是 Qt 框架中的一个控件&#xff0c;用于创建一个可折叠的“工具箱”界面&#xff08;类似 Windows 资源管理器的侧边栏&#xff09;。每个子项可以展开或折叠&#xff0c;适合用于分组显示多个功能模块。以下是其基本用法和示例&#xff1a; 1. 基本用法 创建并添加…

《DeepSeek 一站式工作生活 AI 助手》

最近国产AI工具DeepSeek在全球火出圈&#xff0c;登顶多个国家应用商店&#xff0c;下载量一路飙升。这匹AI “黑马” 到底凭什么征服全球用户&#xff1f;让我们全方位解锁DeepSeek——从基础入门到高阶玩法&#xff0c;从实用技巧到隐藏功能。 DeepSeek是一款功能强大的国产A…

Java中CompletableFuture异步工具类

参考&#xff1a;CompletableFuture 详解 | JavaGuide 实际项目中&#xff0c;一个接口可能需要同时获取多种不同的数据&#xff0c;然后再汇总返回&#xff0c;举个例子&#xff1a;用户请求获取订单信息&#xff0c;可能需要同时获取用户信息、商品详情、物流信息、等数据。…

Oracle Rac 多路径链路不稳定引发IO降速-光弱

一、背景 今天突然被异地的同事拉来开远程会议&#xff0c;会议内容是开发反馈每天9点左右有个sqlldr 命令的脚本调用突然执行很慢&#xff0c;以前几秒的导入操作现在需要30-60s左右&#xff0c;而且数据量基本相同。 二、分析 1&#xff09;、查看ASH报告 从报告上确认是数…

哈希表-两个数的交集

代码随想录-刷题笔记 349. 两个数组的交集 - 力扣&#xff08;LeetCode&#xff09; 内容: 集合的使用 , 重复的数剔除掉&#xff0c;剩下的即为交集&#xff0c;最后加入数组即可。 class Solution {public int[] intersection(int[] nums1, int[] nums2) {Set<Integer…

[JVM篇]分代垃圾回收

分代垃圾回收 分代收集法是目前大部分 JVM 所采用的方法&#xff0c;其核心思想是根据对象存活的不同生命周期将内存划分为不同的域&#xff0c;一般情况下将 GC 堆划分为老生代(Tenured/Old Generation)和新生代(Young Generation)。老生代的特点是每次垃圾回收时只有少量对象…

汉诺塔问题详解:递归与分治的经典案例

嘿&#xff0c;小伙伴们&#xff01;今天我可算撞见了个超有意思的东西&#xff0c;就是那大名鼎鼎的汉诺塔问题&#xff01;我这好奇心一下子就被勾起来了&#xff0c;迫不及待地想深挖一下&#xff0c;然后把那些好玩的、烧脑的、让人拍案叫绝的解题思路和奇妙故事都分享给大…

vue中如何动态的增减组件的类名(class)

在 Vue.js 2 中&#xff0c;你可以通过计算属性或直接在模板中使用 v-bind:class 来动态地改变组件的类名。下面是一个简单的示例&#xff0c;说明如何在某个条件被复核后为组件添加一个 selected 类&#xff08;此处为组件添加一个默认的类&#xff08;例如 radio&#xff09;…

Vue3 基础概念与环境搭建

一、Vue3 简介 Vue3 是 Vue.js 的最新主要版本&#xff0c;于 2020 年 9 月正式发布。它在性能、可维护性和开发体验方面都有了显著的改进。相比 Vue2&#xff0c;Vue3 的主要特点包括&#xff1a; 更高效的响应式系统&#xff1a;使用 Proxy替代了 Object.defineProperty&…

华为昇腾920b服务器部署DeepSeek翻车现场

最近到祸一台HUAWEI Kunpeng 920 5250&#xff0c;先看看配置。之前是部署的讯飞大模型&#xff0c;发现资源利用率太低了。把5台减少到3台&#xff0c;就出了他 硬件配置信息 基本硬件信息 按照惯例先来看看配置。一共3块盘&#xff0c;500G的系统盘&#xff0c; 2块3T固态…

Python的那些事第二十三篇:Express(Node.js)与 Python:一场跨语言的浪漫邂逅

摘要 在当今的编程世界里,Node.js 和 Python 像是两个性格迥异的超级英雄,一个以速度和灵活性著称,另一个则以强大和优雅闻名。本文将探讨如何通过 Express 框架将 Node.js 和 Python 结合起来,打造出一个高效、有趣的 Web 应用。我们将通过一系列幽默风趣的实例和表格,展…

Word中接入大模型教程

前言 为什么要在word中接入大模型呢&#xff1f; 个人觉得最大的意义就是不用来回切换与复制粘贴了吧。 今天分享一下昨天实践的在word中接入大模型的教程。 在word中接入大模型最简单的方式就是使用vba。 vba代码要做的事&#xff0c;拆分一下就是&#xff1a; 获取用户…

open3d绘制平面

在Open3D中绘制平面通常涉及到创建一个平面模型并将其可视化。Open3D是一个开源库,主要用于3D数据的处理和可视化,但它主要用于3D数据的处理,并不直接支持绘制2D平面。如果你想在Open3D中“绘制”一个平面,你可以通过以下几种方法来实现类似的效果: 方法1:使用o3d.geome…

DeepSeek R1 与 OpenAI O1:机器学习模型的巅峰对决

我的个人主页 我的专栏&#xff1a;人工智能领域、java-数据结构、Javase、C语言&#xff0c;希望能帮助到大家&#xff01;&#xff01;&#xff01;点赞&#x1f44d;收藏❤ 一、引言 在机器学习的广袤天地中&#xff0c;大型语言模型&#xff08;LLM&#xff09;无疑是最…

WebGPU顶点插槽进阶优化指南:释放GPU渲染性能

本文基于WebGPU官方规范与实践经验&#xff0c;深入探讨顶点缓冲区的性能优化策略&#xff0c;涵盖数据布局、资源管理、渲染流程等多个维度&#xff0c;并附详细代码注释与性能对比分析。 一、数据布局优化&#xff1a;降低内存与带宽压力 1. 内存对齐策略 GPU对内存访问有严…

数据结构实现顺序表的尾插,尾删,按值查找/修改/删除,按下标查找/增加/删除

头文件&#xff1a;head.h #ifndef __HEAD_H__ #define __HEAD_H__#include <stdio.h> #include <string.h> #include <stdlib.h> #define MAXSIZE 20enum num {success,false-1};typedef int datatype;typedef struct {int len;datatype data[MAXSIZE]; }S…

基于Spring Boot+Vue的宠物服务管理系统(源码+文档)

项目简介 宠物服务管理系统实现了以下功能&#xff1a; 基于Spring BootVue的宠物服务管理系统的主要使用者分为用户管理模块&#xff0c;由于系统运行在互联网络中&#xff0c;一些游客或者病毒恶意进行注册&#xff0c;产生大量的垃圾用户信息&#xff0c;管理员可以对这些…

2. grafana插件安装并接入zabbix

一、在线安装 如果不指定安装位置&#xff0c;则默认安装位置为/var/lib/grafana/plugins 插件安装完成之后需要重启grafana 命令在上一篇讲到过 //查看相关帮助 [rootlocalhost ~]# grafana-cli plugins --help //从列举中的插件过滤zabbix插件 [rootlocalhost ~]# grafana…

【Linux】Ubuntu Linux 系统——Python集成开发环境

ℹ️大家好&#xff0c;我是练小杰&#xff0c;今天周四了&#xff0c;明天就周五了&#xff0c;再坚持坚持又能休息了&#xff01;&#xff01;&#x1f606; 本文是有关Linux 操作系统中Python集成开发环境基础知识&#xff0c;后续将添加更多相关知识噢&#xff0c;谢谢各位…

DeepSeek+即梦 做AI视频

DeepSeek做AI视频 制作流程第一步&#xff1a;DeepSeek 生成视频脚本和分镜 第二步&#xff1a;生成分镜图片绘画提示词第三步&#xff1a;生成分镜图片第四步&#xff1a;使用可灵 AI 工具&#xff0c;将生成的图片转成视频。第五步&#xff1a;剪映成短视频 DeepSeek 真的强&…