【剑指Offer】斐波那契数列——两种解法

本文介绍了解决斐波那契数列问题的两种方法:一种是简单的递归方法,时间复杂度为O(2^n),另一种是采用动态规划方法,将时间复杂度降低到O(N)。此外还提到了更高效的矩阵乘法方式,其时间复杂度为O(logN)。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

斐波那契数列——两种解法

题目描述:

给定整数N,返回斐波那契数列的第N项。

n0-1-2-3-4-5-6-7-8
Fib(n)01-12-35-813-21

解题思路

首先是O(2^n)的解法,除第一项和第二项以外,如果N为正整数,则F(n)=F(n-2)+F(n-1),如果N为负整数,则F(n) = F(n-2)-F(n-1)。于是暴力递归的方法如下:

    /**
     * 使用递归的方法
     *
     * @param n           要计算的数字
     * @param isPositive: 是否是正数,如果是,则为true, 否则为false
     * @return
     */
    private static BigInteger fib_1(int n, boolean isPositive) {
        if (n == 0) {
            return BigInteger.valueOf(0);
        } else if (n == 1 || n == -1) {
            return BigInteger.valueOf(1);
        } else if (isPositive) {
            return fib_1(n - 2, isPositive).add(fib_1(n - 1, isPositive));
        } else {
            return fib_1(n + 2, isPositive).subtract(fib_1(n + 1, isPositive));
        }
    }

使用暴力递归的方法没有记录每个中间数值,因此有很多重复计算的问题,所以,我们考虑使用动态规划的方法,记录每个中间数值,时间复杂度为O(N)。代码如下:

    /**
     * 使用动态规划的方法
     *
     * @param n:                要计算的数字
     * @param isPositive:是否为正整数
     * @return
     */
    private static BigInteger fib_2(int n, boolean isPositive) {
        if (n == 0) {
            return BigInteger.valueOf(0);
        }
        if (n == 1 || n == -1) {
            return BigInteger.valueOf(1);
        }
        // 使用Math的abs函数,计算n的绝对值,将其作为构建动态规划数组的大小
        int size = Math.abs(n);
        // 创建动态规划数组,记录每个阶段的数值
        BigInteger[] dp = new BigInteger[size + 1];
        dp[0] = BigInteger.valueOf(0);// 第一个位置为0
        dp[1] = BigInteger.valueOf(1);// 第二个位置为1
        for (int i = 2; i <= size; i++) {
            if (isPositive) {
                dp[i] = dp[i - 2].add(dp[i - 1]);
            } else {
                dp[i] = dp[i - 2].subtract(dp[i - 1]);
            }
        }
        return dp[size];
    }

当然,还有其他时间复杂度更低的方法,比如使用矩阵乘法的方式,可以做到时间复杂度为O(logN)。有兴趣的读者可以自行研究。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

镰刀韭菜

看在我不断努力的份上,支持我吧

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值