目录

什么是斐波那契数列?

用递归推导Fibonacci

复杂度分析

 用迭代推导Fibonacci

 复杂度分析

 递归优化:记忆化递归(Memoized Recursion) 

复杂度分析


什么是斐波那契数列?

斐波那契数列(Fibonacci Sequence)是这样一个序列:

从第 2 项开始,每一项等于前两项之和。

F(0) = 0
F(1) = 1
F(2) = 1        ← 0 + 1
F(3) = 2        ← 1 + 1
F(4) = 3        ← 1 + 2
F(5) = 5        ← 2 + 3
F(6) = 8        ← 3 + 5
F(7) = 13       ← 5 + 8
F(8) = 21       ← 8 + 13
...

用递归推导Fibonacci

我们用自然语言重新表述递归的核心思想:

要计算第 n 项,只需递归调用第 n−1 项和第 n−2 项的计算,直到我们知道结果的那一刻(也就是 n==0 或 n==1)为止。

所以递归的思想是:

  1. 边界条件(Base Case):当 n == 0 或 n == 1,直接返回 n 本身。

  2. 递归调用:否则,就返回 fib(n - 1) + fib(n - 2)

int fib(int n) {if (n <= 1) return n;         // base casereturn fib(n - 1) + fib(n - 2); // recursion
}

复杂度分析

我们以 fib(5) 为例,画出调用树结构 

                      fib(5)/      \fib(4)        fib(3)/     \        /     \fib(3)     fib(2)  fib(2)  fib(1)/     \     /   \   /   \fib(2)   fib(1) fib(1) fib(0) fib(1)/   \
fib(1) fib(0)

📈 时间复杂度 

T(n) = T(n-1) + T(n-2) + O(1)

其中:

  • T(n-1):递归求 fib(n-1)

  • T(n-2):递归求 fib(n-2)

  • O(1):加法操作 + 1 次返回 

 所以 T(n) 的增长趋势就是 Fibonacci 的大小增长趋势,即:

T(n) ≈ O(2ⁿ)

🔍 你可以这么理解这个复杂度

在最差情况下,调用树是一个二叉树

  • 高度:n

  • 节点数 ≈ 2ⁿ(满二叉树)

  • 每个节点代表一次函数调用 → 所以总时间就是 O(2ⁿ)

 故时间复杂度是 O(2ⁿ)

复杂度说明
时间复杂度O(2ⁿ),因为调用树呈指数爆炸,每次展开两个分支
空间复杂度O(n),因为递归栈最深可达 n 层


 用迭代推导Fibonacci

核心问题:如何“推进到下一步”?

我们写下第一项和第二项: 

step 0: 0   → 这是 F(0)
step 1: 1   → 这是 F(1)

从 step 2 开始,每一步我们都:

  1. 把上面两个数加起来,得出当前项

  2. 把这两个数“往前移动”,以备下一次使用

因此,我们要用变量表示这个“移动窗口”,让它们“滑动”到下一项:(有点类似于SQL中的窗口滑动)

我们只要两个变量:

a = 0;   // 表示 F(n - 2)
b = 1;   // 表示 F(n - 1)

我们用第三个变量:

next = a + b;  // 当前项 F(n)

之后我们把窗口向前滑动:

a = b;
b = next;

 这三步形成了一个“更新循环”。

完整代码:

思考逻辑结构

  • 需要处理:

    • n == 0:返回 0

    • n == 1:返回 1

  • i = 2 开始循环,到 i == n

  • 使用变量 a, b, next 来存储 F(n-2), F(n-1), F(n)

#include <iostream>
using namespace std;int fib_iterative(int n) {if (n == 0) return 0;  // 基础情况if (n == 1) return 1;  // 基础情况int a = 0;  // 表示 F(0)int b = 1;  // 表示 F(1)int next;   // 用来存当前项 F(n)for (int i = 2; i <= n; ++i) {next = a + b;  // 当前项a = b;         // 向前滑动b = next;      // 向前滑动}return next;  // 最终 next 是 F(n)
}

 复杂度分析

⏱️时间复杂度(Time Complexity)

看看上面这段代码里,哪些操作会随着 n 增长而重复?

  • for (int i = 2; i <= n; ++i)
    这个循环会执行 (n - 1) 次(从 i=2 到 i=n)

  • 每次循环内,做了以下操作:

    • next = a + b;

    • a = b;

    • b = next;

这些都是常数时间操作,每次循环都执行固定次数。

✅ 结论:

  • 循环执行 n - 1 次,每次是 O(1)

  • 所以:时间复杂度为 O(n)

🗃️空间复杂度(Space Complexity) 

我们在函数中定义了:

  • int a = 0;

  • int b = 1;

  • int next;

无论 n 多大,我们始终只使用这 3 个变量来计算 Fibonacci 数列。

我们没有使用数组、没有递归栈,也没有任何随着 n 增长而变大的数据结构。

✅ 结论:

  • 内存使用是常数级别

  • 所以:空间复杂度为 O(1)


 递归优化:记忆化递归(Memoized Recursion) 

在递归实现中,你会发现一个规律:很多调用是重复的!

例如在fib(5)中

  • fib(3) 被调用了 2 次

  • fib(2) 被调用了 3 次

  • fib(1) 被调用了 5 次

这就是递归 Fibonacci 的最大问题 —— 重叠子问题!这是一种指数级的浪费。

✅ 第一性解法:我们应该记住已经算过的结果。 

💡如何实现“记住”?

我们用一种最基本的数据结构 —— 数组,开一个数组 memo[] 存储结果。

逻辑流程:

我先检查 memo[n] 是否已经存有值:

  • 如果有,就直接返回它;

  • 如果没有,就计算出来,并把结果存进去。

一步一步改造原始递归函数

原始递归(无记忆):

int fib(int n) {if (n == 0) return 0;if (n == 1) return 1;return fib(n - 1) + fib(n - 2);
}

✅ 加入“备忘录数组”(记忆)

const int MAX = 1000;
int memo[MAX];  // 初始化为 -1 表示未计算void init_memo() {for (int i = 0; i < MAX; ++i) {memo[i] = -1;}
}int fib(int n) {if (memo[n] != -1) return memo[n];  // 如果已经算过了if (n == 0) return memo[0] = 0;if (n == 1) return memo[1] = 1;// 没算过,就递归计算,并存入 memomemo[n] = fib(n - 1) + fib(n - 2);return memo[n];
}

刚刚实现的其实是“Top-Down 动态规划”

虽然我们没有提任何术语,但你用的是一种“从上往下递归,同时缓存子结果”的方法。

如果你用迭代而非递归,那就是“Bottom-Up 动态规划”

复杂度分析

项目原始递归记忆化递归
时间复杂度O(2ⁿ)(指数级)O(n)(每个 n 只算一次)
空间复杂度O(n)(递归栈)O(n)
核心优化原理消除重复计算每个子问题只解一次

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/pingmian/87086.shtml
繁体地址,请注明出处:http://hk.pswp.cn/pingmian/87086.shtml
英文地址,请注明出处:http://en.pswp.cn/pingmian/87086.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ArcGIS Pro利用擦除工具,矢量要素消除另一矢量部分区域

选择“System Toolboxes”→“Analysis Tools.tbx”→“Overlay”→“Erase&#xff08;擦除&#xff09;”。 原始 擦除后

Linux: network: 性能 pause

最近看到一个问题,是关于网卡的throughput的性能问题,后来在ethtool-S里看到有pause的counter,这个也是网络性能问题的一个分析方向。算是学到了新的知识点。 $ grep -i -e 2025- -e pause ethtool*ens2f1np1 | grep -v -e ": 0\$" | headtail 4====

目标检测系列(五)已标注数据集(yolo格式)导入labelstudio继续标注

目录 1、labelstudio安装 2、yolo(txt)转json 3、COCO转yolo(仅针对coco格式标注信息) 4、设置环境变量并启动labelstudio 5、进入label studio创建工程并设置任务标签 6、安装http-server并启动文件映射服务 7、进入label studio导入json文件即可 1、labelstudio安装 …

pytorch底层原理学习--Libtorch

libtorch libtorch 是 PyTorch 的 C 实现版本&#xff0c;可以认为所有的pytorch底层都是由c实现&#xff0c;而pytorch的所有C实现就叫libtorch&#xff0c;也就是我们在pytorch官网getstart页面下载的cpytorch版本。我们用python写的pytorch神经网络代码都会通过pybind11将p…

TCP 三次握手协商 MSS 前,如何确定 MSS 值(附 Linux 内核源码)

文章目录 一、SYN总结影响 SYN MSS 的因素 二、SYNACK总结影响 SYNACK MSS 的因素 结合 Linux 内核源码 一、SYN 总结影响 SYN MSS 的因素 套接字选项 TCP_MAXSEG路由选项 advmss出口 MTU 减去 40(TCP 和 IP 的固定首部大小)IPV4_MAX_PMTU - 40(同上) 二、SYNACK 总结影响 SY…

扫描电子显微镜(SEM)夏令营面试基础题及答案

第二期表征问题SEM&#xff0c;后续会陆续更新其他表征 SEM和XRD一样&#xff0c;都是表征里面很常见的手段&#xff0c;基本上看论文这两个都是必不可少的 对于这部分内容&#xff0c;理解记忆&#xff1e;死记硬背&#xff0c;到时会问起来回答个大概就行&#xff0c; 像上…

Leetcode力扣解题记录--第49题(map)

题目链接&#xff1a;49. 字母异位词分组 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给你一个字符串数组&#xff0c;请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。 示例 1: 输入: strs ["eat", "tea", "tan", &quo…

AI赋能智慧餐饮:Spring Boot+大模型实战指南

⚡ 餐饮行业三大痛点 高峰期点餐拥堵&#xff1a;300人餐厅&#xff0c;15个服务员仍排长队 后厨浪费严重&#xff1a;食材损耗率高达25%&#xff0c;成本失控 顾客体验同质化&#xff1a;复购率不足30% &#x1f680; 智慧餐饮解决方案架构 &#x1f525; 核心模块代码实现…

用鸿蒙打造真正的跨设备数据库:从零实现分布式存储

网罗开发 &#xff08;小红书、快手、视频号同名&#xff09; 大家好&#xff0c;我是 展菲&#xff0c;目前在上市企业从事人工智能项目研发管理工作&#xff0c;平时热衷于分享各种编程领域的软硬技能知识以及前沿技术&#xff0c;包括iOS、前端、Harmony OS、Java、Python等…

【Docker基础】Docker数据卷:数据卷的作用与使用场景

目录 1 Docker数据卷概述 1.1 什么是数据卷 1.2 数据卷的核心特性 3 数据卷与绑定挂载的对比 2.1 技术对比 2.2 选择建议 3 数据卷的核心作用 3.1 数据持久化 3.2 数据共享 3.3 备份与迁移 4 数据卷使用场景详解 4.1 数据库应用 4.2 日志集中管理 5 数据卷操作全…

安装GPU版本的Pytorch

前言 Pytorch是深度学习框架,在工作中我们一般是使用GPU版本的Pytorch,提高运行效率 安装GPU版本的Pytorch需要先安装CUDA和CUANN这两个GPU环境 如果准备安装GPU版本的Pytorch安装同志没有安装CUDA和CUANN,请看我上一篇文章 RTX5070显卡安装CUDA和CUDNN-CSDN博客 目录 安装…

微信小程序学习笔记

微信小程序学习笔记 一、文件和目录结构介绍 小程序包括&#xff1a;主体文件、页面文件 主体文件&#xff1a; app.js&#xff1a;小程序入口文件app.json&#xff1a;小程序的全局配置文件app.wxss&#xff1a;小程序的全局样式 页面文件&#xff1a;是每个页面所需的文…

抓包之通过wireshark抓ping包

写在前面 本文看下如何抓ping包。 1&#xff1a;正文 因为ping使用的是icmp协议&#xff0c;所以这里我们可以通过过滤icmp协议来进行抓包&#xff1a; 其中对于icmp请求报文状态码是8&#xff0c;如下&#xff1a; 响应状态码是0&#xff1a; 如下图是一个局域网环境中…

大文件分片上传 — nodejs

上传文件路由&#xff1a; var express require(express); var router express.Router(); const multer require(multer); const fs require(fs); const path require(path);// 确保上传目录存在 const uploadDir path.join(__dirname, ../backend/uploads); const temp…

HarmonyOS File和base64字符串转换

1. HarmonyOS File和base64字符串转换 1.1. Base64 1.1.1. Base64认知 Base64 是一种基于64个 ASCII 字符来表示二进制数据的表示方法&#xff0c;这个64个不同的字符为&#xff1a;   &#xff08;1&#xff09;大、小写字母&#xff08;A– Z、a–z&#xff09;。52个  …

【NodeJs】【npm】npm安装electron报错

解决问题 npm安装electron报错一般来说是镜像源的问题。 electron的镜像源与一般的 vue 之类的镜像源地址不一样需要单独配置。 npm读取的全局配置一般是在 C:\Users\{用户}\.npmrc 这个配置文件中。 如果你找不到你的配置文件可以执行如下命令, # 执行后会直接用txt打开你的…

植物small RNA靶基因预测软件,psRobot

psRoto软件安装 网址 http://omicslab.genetics.ac.cn/psRobot/downloads.php下载和安装 wget http://omicslab.genetics.ac.cn/psRobot/program/WebServer/psRobot_v1.2.tar.gz # tar -zxvf psRobot_v1.2.tar.gz # cd psRobot_v1.2 ## ./configure make make installpsRot…

翻译服务器

基于UDP编程博客里的回显服务器代码,翻译服务只需要改process方法即可 所以我们可以创建一个UdpDictServer直接继承UdpEchoServer然后重写process方法 在重写的方法中完成翻译的过程 代码: package network;import java.io.IOException; import java.net.SocketException; …

初等变换 线性代数

初等变换 介绍了三种初等变换的操作。 初等矩阵 初等矩阵是干嘛的呢&#xff1f;实际上初等矩阵就是我们矩阵的初等操作&#xff0c;每一个对矩阵的初等变换操作都相当于乘上一个初等矩阵。 左乘初等矩阵就相当于对行进行初等操作&#xff0c;右乘则相当于对列进行初等操作。…

Java基础 集合框架 队列架构 双端队列 Deque

双端队列 Deque Deque 方法简介Deque 核心特点Deque实现类 ArrayDequeArrayDeque 构造方法ArrayDeque 的数据结构及实现原理ArrayDeque 方法介绍ArrayDeque 核心特性ArrayDeque 总结ArrayDeque 使用样例代码 Deque实现类 LinkedListDeque实现类 ConcurrentLinkedDeque (非阻塞线…