斐波那契数【C++】【多解法】

题目链接:

题目
斐波那契数 (通常用 F(n) 表示)形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始,后面的每一项数字都是前面两项数字的和。也就是:
F(0) = 0,F(1) = 1
F(n) = F(n - 1) + F(n - 2),其中 n > 1
给定 n ,请计算 F(n) 。
示例 1:
输入:n = 2
输出:1
解释:F(2) = F(1) + F(0) = 1 + 0 = 1
示例 2:
输入:n = 3
输出:2
解释:F(3) = F(2) + F(1) = 1 + 1 = 2
示例 3:
输入:n = 4
输出:3
解释:F(4) = F(3) + F(2) = 2 + 1 = 3
提示:
0 <= n <= 30

解法一:递归

class Solution {
public:
    int Fibonacci(int n) {  //递归参数
        //终止条件
        if (n <= 1)    
             return n;
        else{
            //根据公式递归调用函数
            return Fibonacci(n - 1) + Fibonacci(n - 2); 
        }
        return 0;
    }
};

时间复杂度O(2^n):计算个数是一颗二叉树
空间复杂度O(n):递归栈深度

解法二:记忆化搜索

因为递归法重复计算了很多值,如果把已经计算过的值保留下来就能优化算法。
怎么保存计算过的值?可以使用map,但本题使用数组更简单:

class Solution {
public:
    int f[50]{0};   //保存计算过的值
    int Fibonacci(int n) {     //递归参数
        if (n <= 2) return 1;   //终止条件
        if (f[n] > 0) return f[n];    //记忆化,防止重复计算
        return f[n] = (Fibonacci(n-1)+Fibonacci(n-2));  //递归同时保存已经计算过的值
    }
};

时间复杂度O(n):从0到n没有重复计算
空间复杂度O(n):递归栈的空间

解法三:动态规划(无状态压缩)

  1. 确定dp[i]数组含义:第i个斐波那契值
vector<int> dp(n + 1, 0);
  1. 确定递推公式
dp[i] = dp[i - 1] + dp[i - 2]
  1. 初始化:根据题目描述
dp[0] = 0;
dp[1] = 1;
  1. 遍历顺序:从小到大遍历
for (int i = 2; i <= n; i++){
    dp[i] = dp[i - 1] + dp[i - 2];
}

整体代码:

class Solution {
public:
    int fib(int n) {
        vector<int> dp(n + 1);
        if (n <= 1){
            return n;
        }
        dp[0] = 0;
        dp[1] = 1;
        for (int i = 2; i <= n; i++){
            dp[i] = dp[i - 1] + dp[i - 2];
        }
        return dp[n];
    }
};

时间复杂度O(n):遍历
空间复杂度O(n):dp数组空间

解法四:动态规划(状态压缩)

我们只使用了三个变量,dp[i]、dp[i - 1]、dp[i - 2],因此可以滚动使用三个变量。

class Solution {
public:
    int fib(int n) {
        if (n < 2){
            return n;
        }
        int a = 0;
        int b = 1;
        int sum;
        for (int i = 2; i <= n; i++){
            sum = a + b;
            a = b;
            b = sum;
        }
        return sum;
    }
};

时间复杂度O(n)
空间复杂度O(1):只使用了常数个变量

总结

递归法:从上到下计算
动态规划:从下到上计算

  • 4
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值