题目描述
一只青蛙一次可以跳上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)