JZ8青蛙跳台阶

题目描述
一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法(先后次序不同算不同的结果)。

描述

此题和斐波拉契数列做法一样。也将用三个方法来解决,从入门到会做。
考察知识:递归,记忆化搜索,动态规划和动态规划的空间优化
难度:一星

题解

方法一:递归

题目分析,这是一道经典的递推题目,你可以想如果青蛙当前在第n级台阶上,那它上一步是在哪里呢?
显然,由于它可以跳1级台阶或者2级台阶,所以它上一步必定在第n-1,或者第n-2级台阶,也就是说它跳上n级台阶的跳法数是跳上n-1和跳上n-2级台阶的跳法数之和。
那么初始条件了,f[0] = f[1] = 1。
所以就变成了:f[n] = f[n-1] + f[n-2], 初始值f[0]=1, f[1]=1,目标求f[n]
看到公式很亲切,代码秒秒钟写完。

public class Solution {
    public int JumpFloor(int target) {
        if(target <= 1){
            return 1;
        } 
        return JumpFloor(target-1) + JumpFloor(target-2);
    }
}

优点,代码简单好写,缺点:慢,会超时
时间复杂度:O(2^n)
空间复杂度:递归栈的空间

方法二:记忆化搜索

拿求f[5] 举例
通过图会发现,方法一中,存在很多重复计算,因为为了改进,就把计算过的保存下来。
那么用什么保存呢?一般会想到map, 但是此处不用牛刀,此处用数组就好了。

public class Solution {
    public int JumpFloor(int target) {
       int[] jf = new int[target + 1];
       jf[0] = 1;
       jf[1] = 1;
        
        for(int i = 2; i <= target; i++){
            jf[i] = jf[i-1] +jf[i-2]; 
        }
        return jf[target];
    }
}

时间复杂度在这里插入代码片:O(n), 没有重复的计算
空间复杂度:O(n)和递归栈的空间

方法三:动态规划:自底向上,使用迭代

虽然方法二可以解决此题了,但是如果想让空间继续优化,那就用动态规划,优化掉递归栈空间。
方法二是从上往下递归的然后再从下往上回溯的,最后回溯的时候来合并子树从而求得答案。
那么动态规划不同的是,不用递归的过程,直接从子树求得答案。过程是从下往上。

public class Solution {
    public int JumpFloor(int target) {
        if(target == 0){
            return 1;
        }
        if(target == 1){
            return 1;
        }
        int result = 0;
        int step1 = 1;
        int step2 = 1;
        for(int i = 2; i <= target; i++){
            result = step1 + step2;
            step2 = step1;
            step1 = result;
        }
        return result;
    }
}

时间复杂度:O(n)
空间复杂度:O(n)

你可能感兴趣的:(剑指offer)