【LeetCode】跳跃游戏
给定一组非负整数,初始时处于数组的第一位下标 0 的位置,数组的每个元素代表那个位置可以跳跃的最大长度。判断你是否能够到达数组的最后一位下标。
e.g.
A = [2, 3, 1, 1, 4],返回 true。
A = [3, 2, 1, 0, 4],返回 false。
我的想法是递归
方法一:
bool canJump(vector<int>& nums) {
return jump(nums, );
}
bool jump(vector<int> &nums, int m) {
int last = nums.size() - ;
if (m == last) return true;
int p = min(last, m + nums[m]);
for (int i = p; i > m; i--) { // 若不定义 p,i 初始为 m + nums[m] 并不会造成数组越界,但变量 p 减少了不必要的递归
if (jump(nums, i))
return true;
}
return false;
}
答案称这种方法是递归回溯法,时间复杂度为 O(2n),数据量很大时会超时。

答案提供了多种解法
方法二:
自顶向下动态规划法(优化的回溯法)
作如下定义:如果从数组中某一位置作为起始点,最终能够到达最后一个下标处,则把这个位置称为 “ Good Index ”,否则成为 “ Bad Index ”。因此这个跳跃问题就变成了判断 0 下标是不是一个 “ Good Index ”。
用一个 memo 数组存储原数组每个下标是好的还是坏的,memo 数组元素的值是 GOOD、BAD、UNKNOWN 之一。
e.g. 对于 nums = [2, 4, 2, 1, 0, 2, 0],
| Index | 0 | 1 | 2 | 3 | 4 | 5 | 6 |
| nums | 2 | 4 | 2 | 1 | 0 | 2 | 0 |
| memo | G | G | B | B | B | G | G |
步骤:
- 初始时,memo 中所有元素都是 UNKNOWN,除了最后一个元素是 GOOD。
- 将回溯法中递归的第一步改为 “ 判断当前下标是否 UNKNOWN ”
如果是 KNOWN,根据其值是 GOOD / BAD 返回 true / false;
否则继续执行回溯。 - 一旦知道当前下标是好的还是坏的,将相应的值存在 memo 数组中。
C++实现:
enum Index {
GOOD, BAD, UNKNOWN
};
vector<Index> memo;
bool canJump(vector<int>& nums) {
memo.reserve(nums.size());
for (int i = ; i < nums.size() - ; i++) {
memo[i] = UNKNOWN;
}
memo[nums.size() - ] = GOOD;
return jump(nums, );
}
bool jump(vector<int> &nums, int m) {
if (memo[m] != UNKNOWN)
return memo[m] == GOOD ? true : false;
int p = min((int)nums.size() - , m + nums[m]);
for (int i = p; i > m; i--) {
if (jump(nums, i)) {
memo[m] = GOOD;
return true;
}
}
memo[m] = BAD;
return false;
}
对于数组中每个位置 i,我们在其右边 nums[i] 个元素中寻找 “ Good Index ”,因此这种方法的时间复杂度是 O(n2)。
方法三:
自底向上动态规划法
将自顶向下 DP 的递归消除后就变成了自底向上 DP,这样就不会造成方法栈的过度开销。消除递归的方法是从数组最右边(倒数第二位)往左不断地递推 memo 数组的值。
C++实现:
enum Index {
GOOD, BAD, UNKNOWN
};
vector<Index> memo;
bool canJump(vector<int>& nums) {
memo.reserve(nums.size());
for (int i = ; i < nums.size() - ; i++) {
memo[i] = UNKNOWN;
}
memo[nums.size() - ] = GOOD;
for (int i = nums.size() - ; i >= ; i--) {
int p = min((int)nums.size() - , i + nums[i]);
for (int j = p; j > i; j--) {
if (memo[j] == GOOD) {
memo[i] = GOOD;
break;
}
}
}
return memo[] == GOOD;
}
由于两种 DP 的原理相同,这种方法的时间复杂度也是 O(n2)。由于没有使用递归,终于 Accepted 了,但效率极低。
最优方法:
贪心法
自底向上 DP 中,从后向前递推 memo 数组时,对于每一个下标 i,我们想知道是否能从这个位置到达一个 “ Good Index ”,而当我们找到了从 i 下标能到达的下标范围 ( i, p ] 内最右边的仅仅一个 “ Good Index ”(也可以选择最左边的一个)就把 i 设为 “ Good Index ”。因此,可以把从后向前追踪的一系列 “ Good Index ” 只用一个单独的变量存储,而不是记录在 memo 数组中。
C++实现:
bool canJump(vector<int>& nums) {
int last = nums.size() - ;
for (int i = last; i >= ; i--) {
if (i + nums[i] >= last)
last = i;
}
return last == ;
}
也可以从前向后
bool canJump(vector<int>& nums) {
int n = nums.size(), reach = ; // reach 表示从 0 能到达的最远的下标
for (int i = ; i < n && i <= reach; i++) {
reach = max(reach, i + nums[i]);
if (reach >= n - )
return true;
}
return false;
}
【LeetCode】跳跃游戏的更多相关文章
- Leetcode 跳跃游戏 II
题目链接:https://leetcode-cn.com/problems/jump-game-ii/ 题目大意: 略. 分析: 贪心 + DP. 代码如下: class Solution { pub ...
- LeetCode:跳跃游戏【55】
LeetCode:跳跃游戏[55] 题目描述 给定一个非负整数数组,你最初位于数组的第一个位置.数组中的每个元素代表你在该位置可以跳跃的最大长度.判断你是否能够到达最后一个位置. 示例 1: 输入: ...
- [LeetCode] 45. Jump Game II 跳跃游戏 II
Given an array of non-negative integers, you are initially positioned at the first index of the arra ...
- 力扣Leetcode 45. 跳跃游戏 II - 贪心思想
这题是 55.跳跃游戏的升级版 力扣Leetcode 55. 跳跃游戏 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 你的目标是使用最少的跳跃 ...
- LeetCode 45跳跃游戏&46全排列
原创公众号:bigsai,回复进群加入力扣打卡群. 昨日打卡:LeetCode 42字符串相乘&43通配符匹配 跳跃游戏 题目描述: 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中 ...
- 【LeetCode每天一题】Jump Game II(跳跃游戏II)
Given an array of non-negative integers, you are initially positioned at the first index of the arra ...
- LeetCode(45): 跳跃游戏 II
Hard! 题目描述: 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 你的目标是使用最少的跳跃次数到达数组的最后一个位置. 示例: 输入: [ ...
- [Leetcode]44.跳跃游戏Ⅰ&&45.跳跃游戏Ⅱ
跳跃游戏链接 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 判断你是否能够到达最后一个位置. 示例 1: 输入: [2,3,1,1,4] 输出 ...
- Leetcode力扣45题 跳跃游戏 II
原题目: 跳跃游戏 II 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素代表你在该位置可以跳跃的最大长度. 你的目标是使用最少的跳跃次数到达数组的最后一个位置. 示例: 输入: ...
- LeetCode 45. 跳跃游戏 II | Python
45. 跳跃游戏 II 题目来源:https://leetcode-cn.com/problems/jump-game-ii 题目 给定一个非负整数数组,你最初位于数组的第一个位置. 数组中的每个元素 ...
随机推荐
- 解析Django路由层URLconf
目录: 一 Django中路由的作用 二 路由的分组 三 路由分发 四 反向解析 五 名称空间 六 Django2.0版的path 一.Django中路由的作用 URL配置(URLconf ...
- WebGIS前端地图显示之根据地理范围换算出瓦片行列号的原理(核心)
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 1.前言 在上一节中我们知道了屏幕上一像素等于实际中多少单位长度(米或 ...
- IIS发布静态页面配置
第一步:按照正常网站发布添加网站: 第二步:修改该网站的默认文档: 第三步:添加默认文档,把静态页的名称添加进去: 第四步:重启网站,浏览:
- Mongodb 创建管理员帐号与普通帐号
数据库操作权限 readAnyDatabase 任何数据库的只读权限 userAdminAnyDatabase 任何数据库的读写权限 userAdminAnyDatabase 任何数据库用户的管理权限 ...
- ZZNU 2095 : 我只看看不写题
把所有时间加起来,最后从大到小排序,一定要把大的先减去.注意花费的时间都是1,这一秒用过就不能再用了,所有用到了并查集的部分知识 #include<iostream> #include&l ...
- SpringBoot之profile的使用
Profile配置是针对不同的环境提供不同的配置支持,比如,在开发环境的配置和测试环境下的配置不同,那么就可以使用Profile配置来实现该要求. 在你的src/main/resources下建立相应 ...
- 使用Hexo搭建一个简单的博客(二)
昨天想着用Hexo和github搭一个自己简单的博客,记录一下自己踩过的坑,具体的流程就不重复了,主要参考了一下几篇文章 GitHub+Hexo 搭建个人网站详细教程 使用Hexo+Github一步步 ...
- d3 .each()
d3.select("xxx") .each(function (d) { //this表示当前element 而 d表示绑定的数据 something(this); }); 注意 ...
- maven项目依赖jar包报 java.lang.classnotfoundexception:Type com.xx.xx.xxx not present 的解决
今天在工作的时候遇到了这样一个奇葩的异常: java.lang.classnotfoundexception:Type com.ys.yahu.vo.file.MobileFileVo not pre ...
- OP社区相关
●相关网站官网: http://openstack.org Wiki: http://wiki.openstack.org 代码贡献统计:http://stackalytics.com/ Bug跟踪: ...