重点看 116. 跳跃游戏(动态规划、贪心)、117. 跳跃游戏 II(贪心)

116. 跳跃游戏

给出一个非负整数数组,你最初定位在数组的第一个位置。数组中的每个元素代表你在那个位置可以跳跃的最大长度。    

判断你是否能到达数组的最后一个位置。

 注意事项

这个问题有两个方法,一个是贪心和 动态规划

贪心方法时间复杂度为O(N)

动态规划方法的时间复杂度为为O(n^2)

我们手动设置小型数据集,使大家可以通过测试的两种方式。这仅仅是为了让大家学会如何使用动态规划的方式解决此问题。如果您用动态规划的方式完成它,你可以尝试贪心法,以使其再次通过一次。

动态规划:针对每一个点都最优

贪心:以最优解决问题,不必每个点最优

样例

A = [2,3,1,1,4],返回 true.

A = [3,2,1,0,4],返回 false.

1、动态规划(时间复杂度为O(n^2),时间复杂度虽大,但需要掌握此方法)

    每到一个点 i,我们扫描之前所有的点,如果之前某点j本身可达,并且与current 点可达,表示点i是可达的。

    返回值: 数组的最后一个值。

class Solution {
public:
    /*
     * @param A: A list of integers
     * @return: A boolean
     */
    bool canJump(vector<int> &A) {
        // write your code here
     
        bool can[A.size()];
        can[0] = true;
        
        for (int i = 1; i < A.size(); i++) 
        {
            for (int j = 0; j < i; j++) 
            {
                if (can[j] && j + A[j] >= i) 
                {
                    can[i] = true;
                    break;
                }
            }
        }
        
        return can[A.size()-1];
    }
};
2、贪心算法

其实这题最好的解法不是DP,而是贪婪算法Greedy Algorithm,因为我们并不是很关心每一个位置上的剩余步数,我们只希望知道能否到达末尾,也就是说我们只对最远能到达的位置感兴趣,所以我们维护一个变量reach,表示最远能到达的位置,初始化为0。遍历数组中每一个数字,如果当前坐标大于reach或者reach已经抵达最后一个位置则跳出循环,否则就更新reach的值为其和i + nums[i]中的较大值,其中i + nums[i]表示当前位置能到达的最大位置,代码如下:

class Solution {
public:
    bool canJump(vector<int>& nums) {
        int n = nums.size(), reach = 0;
        for (int i = 0; i < n; ++i) {
            if (i > reach || reach >= n - 1) break;
            reach = max(reach, i + nums[i]);
        }
        return reach >= n - 1;
    }
};

117. 跳跃游戏 II

给出一个非负整数数组,你最初定位在数组的第一个位置。数组中的每个元素代表你在那个位置可以跳跃的最大长度。   

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

样例

给出数组A = [2,3,1,1,4],最少到达数组最后一个位置的跳跃次数是2(从数组下标0跳一步到数组下标1,然后跳3步到数组的最后一个位置,一共跳跃2次)

思想: 
此题默认肯定可以到达末尾的。
与上题不同的是,我们需要找到在每一步跳过之后的下一步里能跳的最大范围,比如第一步从0可以最大跳到2,那么我们就寻找从位置1和2中能跳的最大距离,这样选出来,才能保证步数最少。


我们需要两个变量cur和pre分别来保存当前的能到达的最远位置和之前能到达的最远位置,只要cur未达到最后一个位置则循环继续,pre先赋值为cur的值,表示上一次循环后能到达的最远位置,如果当前位置i小于等于pre,说明还是在上一跳能到达的范围内,我们根据当前位置加跳力来更新cur,更新cur的方法是比较当前的cur和i + A[i]之中的较大值,如果题目中未说明是否能到达末尾,我们还可以判断此时pre和cur是否相等,如果相等说明cur没有更新,即无法到达末尾位置,返回-1,代码如下:

 
  
class Solution {
public:
 int jump(vector<int>& nums) {
 int step = 0, n = nums.size(), cur = 0;      
 while (cur < n - 1) 
{ ++step; 
  int pre = cur;
  for (int i=0; i <= pre; ++i) 
  cur = max(cur, i + nums[i]); 
  if (pre == cur) 
  return -1; // May not need this 
  return step; 
 }
};





猜你喜欢

转载自blog.csdn.net/weixin_41413441/article/details/79164933