斐波那契数列(Fibonacci)


一、背景介绍

斐波那契数列(Fibonacci sequence),又称黄金分割数列,因数学家莱昂纳多·斐波那契(Leonardo Fibonacci)以兔子繁殖为例子而引入,故又称为“兔子数列”。

【兔子繁殖问题】

一般而言,兔子在出生两个月后,就有繁殖能力,一对兔子每个月能生出一对小兔子来。如果所有兔子都不死,那么一年以后可以繁殖多少对兔子?


二、数学定义

指的是这样一个数列:0、1、1、2、3、5、8、13、21、34、……

用数学课表示为:


三、核心代码

方法一:递归

第n个Fibonacci数可递归地计算为:

int fibonacci(int n) {
if (n <= 1)
return 1;
return fibonacci(n - 1) + fibonacci(n - 2);
}

方法二:迭代

int fibonacci(int n) {
if (n <= 1)
return 1;
int f = 1, g = 1,ret = 0;
for (int i = 2; i <= n; i++) {
ret = f + g;
f = g;
g = ret;
}
return ret;
}

四、时间复杂度分析

方法一:递归

即可得到O(n^2)。

方法二:迭代

第一种解法比较简单,但是多个元素重复计算,因而时间复杂度较高,为了避免重复计算,可进行循环计算减少时间复杂度,降为O(n)。

五、完整代码

方法一:递归

#include<iostream>
#define scanf scanf_s
using namespace std; int fibonacci(int n) {
if (n <= 1)
return 1;
return fibonacci(n - 1) + fibonacci(n - 2);
} int main() {
printf("请输入下标:");
int n;
scanf_s("%d", &n);
printf("下标为%d的fibonacci数为%d\n",n, fibonacci(n));
return 0;
}

方法二:迭代

#include<iostream>
#define scanf scanf_s
using namespace std; int fibonacci(int n) {
if (n <= 1)
return 1;
int f = 1, g = 1,ret = 0;
for (int i = 2; i <= n; i++) {
ret = f + g;
f = g;
g = ret;
}
return ret;
} int main() {
printf("请输入下标:");
int n;
scanf_s("%d", &n);
printf("下标为%d的fibonacci数为%d\n",n, fibonacci(n));
return 0;
}

六、提升设计

因而计算f(n)就简化为了计算矩阵的(n-2)次方,而计算矩阵的(n-2)次方,我们又可以进行分解,即计算矩阵(n-2)/2次方的平方,逐步分解下去,由于折半计算矩阵次方,因而时间复杂度为O(log n) 。

#include<iostream>
using namespace std; class Matrix {
public:
int n;
int** m;
Matrix(int num) {
m = new int* [num];
for (int i = 0; i < num; i++) {
m[i] = new int[num];
}
n = num;
clear();
}
void clear() {
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
m[i][j] = 0;
}
}
}
void unit() {
clear();
for (int i = 0; i < n; i++) {
m[i][i] = 1;
}
}
Matrix operator=(const Matrix mtx) {
Matrix(mtx.n);
for (int i = 0; i < mtx.n; ++i) {
for (int j = 0; j < mtx.n; ++j) {
m[i][j] = mtx.m[i][j];
}
}
return *this;
}
Matrix operator*(const Matrix& mtx) {
Matrix result(mtx.n);
result.clear();
for (int i = 0; i < mtx.n; ++i) {
for (int j = 0; j < mtx.n; ++j) {
for (int k = 0; k < mtx.n; ++k) {
result.m[i][j] += m[i][k] * mtx.m[k][j];
}
}
}
return result;
}
}; int main(int argc,const char*argv[]) {
unsigned int num = 2;
Matrix first(num);
first.m[0][0] = 1;
first.m[0][1] = 1;
first.m[1][0] = 1;
first.m[1][1] = 0;
int t;
cout << "请输入下标: ";
cin >> t;
Matrix result(num);
result.unit();
int n = t - 2;
while (n) {
if (n % 2) {
result = result * first;
}
first = first * first;
n = n / 2;
}
cout << (result.m[0][0] + result.m[0][1]) << endl;
return 0;
}

算法1:Fibonacci数列的更多相关文章

  1. c语言经典算法---计算Fibonacci数列

    算法是一个程序和软件的灵魂,作为一名优秀的程序员,只有对一些基础的算法有着全面的掌握,才会在设计程序和编写代码的过程中显得得心应手.下面我就分享一个C语言中比较基础却极为重要的一个算法----计算Fi ...

  2. java算法 蓝桥杯算法训练 Fibonacci数列

    问题描述 Fibonacci数列的递推公式为:Fn=Fn-1+Fn-2,其中F1=F2=1. 当n比较大时,Fn也非常大,现在我们想知道,Fn除以10007的余数是多少. 输入格式 输入包含一个整数n ...

  3. 算法——js(Fibonacci数列)

    斐波那契数列(Fibonacci sequence),又称黄金分割数列,因数学家列昂纳多·斐波那契(Leonardoda Fibonacci[1]  )以兔子繁殖为例子而引入,故又称为“兔子数列”,指 ...

  4. 算法设计与分析 1.2 不一样的fibonacci数列 (矩阵快速幂思想)

    题目描述 Winder 最近在学习 fibonacci 数列的相关知识.我们都知道 fibonacci 数列的递推公式是F(n) = F(n - 1) + F(n - 2)(n >= 2 且 n ...

  5. 程序员面试题精选100题(16)-O(logn)求Fibonacci数列[算法]

    作者:何海涛 出处:http://zhedahht.blog.163.com/ 题目:定义Fibonacci数列如下: /  0                      n=0 f(n)=      ...

  6. 【算法】Fibonacci(斐波那契数列)相关问题

    一.列出Fibonacci数列的前N个数 using System; using System.Collections.Generic; using System.Linq; using System ...

  7. 算法设计与分析 1.2 不一样的fibonacci数列

    ★题目描述 fibonacci 数列的递推公式是F(n) = F(n-1) + F(n-2)(n >= 2 且 n 为整数). 将这个递推式改为F(n) = aF(n-1) + bF(n-2)( ...

  8. fibonacci 数列及其应用

    fibonacci 数列及其延展 fibonacci计算 fibonacci数列是指 0,1,1,2,3,5,8,13,21……这样自然数序列,即从第3项开始满足f(n)=f(n-1)+f(n-2): ...

  9. 蓝桥网试题 java 入门训练 Fibonacci数列

    ---------------------------------------------------------------------------------------------------- ...

  10. Fibonacci数列的解法

    Fibonacci数列的解法: 1.递归算法 递归的概念,我说不清楚,语文不好.但是核心思想,我认为就是入栈出栈.比方说,你想要求得某个结果,如果一步求解不出来,那么先把最后一步的计算步骤进栈,先不考 ...

随机推荐

  1. 使用 Sealos 一键部署高可用 MinIO,开启对象存储之旅

    大家好!今天这篇文章主要向大家介绍如何通过 Sealos 一键部署高可用 MinIO 集群. MinIO 对象存储是什么? 对象是二进制数据,例如图像.音频文件.电子表格甚至二进制可执行代码.对象的大 ...

  2. web应用及微信小程序版本更新检测方案实践

    背景: 随着项目体量越来越大,用户群体越来越多,用户的声音也越来越明显:关于应用发版之后用户无感知,导致用户用的是仍然还是老版本功能,除非用户手动刷新,否则体验不到最新的功能:这样的体验非常不好,于是 ...

  3. 一文搞懂 OTP 双因素认证

    GitHub 在 2023 年 3 月推出了双因素认证(two-factor authentication)简称 2FA,并且承诺所有在 GitHub 上贡献的开发者在 2023 年底前启用双因素认证 ...

  4. Goobye, cnblogs

    转 typecho 了,个人网站的客制化程度当然不是 cnblogs 能比得上的. <cirnovsky.cf>

  5. 【matplotlib基础】--动画

    matplotlib的动画一直是一个强大但使用频率不高的功能,究其原因,一方面展示动画需要一定的媒介,没有图形和文字展示方便:二来大家更关心的是分析结果的最终图表,图表的动态展示则没有那么重要. 不过 ...

  6. Redis——Redis面试题

    文章目录 概述 什么是Redis Redis有哪些优缺点 为什么要用 Redis /为什么要用缓存 为什么要用 Redis 而不用 map/guava 做缓存? Redis为什么这么快 数据类型 Re ...

  7. 解决在VS Code中运行有中文字符的Java代码(第三种方式),出现编码 GBK 的不可映射字符 (0x81)

    写代码时,我们不避免的会使用一些中文注释,这些在其他的语言中没有问题.但是在Java的注释里面如果有中文字符,就会报错.即使文件编码是utf-8也无济于事.是因为使用CMD运行java程序的时候,系统 ...

  8. SQLPLUS使用及Oracle表空间设定自动扩展

    起因:ERP不能登陆,Oracle无法访问,报错如下 后联系鼎捷se提供以下解决方案: 该问题是由于Oracle审计表AUD$数据过大导致数据库异常,现已登录DB服务器使用oracle账号执行语句tr ...

  9. 使用 GitHub Action 自动更新 Sealos 集群的应用镜像

    在 IT 领域,自动化无疑已成为提高工作效率和减少人为错误的关键.Sealos 作为一个强大的云操作系统,已经为许多企业和开发者提供了稳定可靠的服务.与此同时,随着技术不断发展,集成更多的功能和服务变 ...

  10. 手撕Vue-编译指令数据

    经过上一篇的分析,完成了查找指令和模板的功能,接下来就是编译指令的数据了. 所以本章节主要处理的方法则是 buildElement 方法,我们先分析一下我们所拿到的数据在进行编码,这样会更加清晰一些. ...