斐波那契数列的--------5种算法(又称“兔子数列”)

news2025/1/10 23:53:11

斐波那契数列(Fibonacci sequence),又称黄金分割数列,因数学家莱昂纳多·斐波那契(Leonardo Fibonacci)以兔子繁殖为例子而引入,故又称为“兔子数列”,指的是这样一个数列:1、1、2、3、5、8、13、21、34、……在数学上,斐波那契数列以如下被以递推的方法定义:F(0)=0,F(1)=1, F(n)=F(n - 1)+F(n - 2)(n ≥ 2,n ∈ N*)

一、循环法:
// n 太大会溢出,可以使用数据位更宽的数据类型
int fib(int n){
	if(n==1||n==2)  return 1;
	int f1=1;  
	int f2=1;  
	int res=0;  //存放结果 
	for(int i=3;i<=n;++i){
		res=f1+f2;
		f2=f1;
		f1=res;
	}
	return res;	
}

时间复杂度:O(n)
空间复杂度:O(1)

二、递归法:

根据递推公式:F(n)=F(n - 1)+F(n - 2)可直接得出

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

在这里插入图片描述

时间复杂度:因为成二叉树形式 ,所以时间复杂度为O(2^n)
空间复杂度:树约有n-1层,所以空间复杂度为O(n)

三、动态规划法:
  1. 确定 dp数组以及下标的含义:dp[i] 的定义为:第 i 个数的斐波那契数值是 dp[i]
  2. 确定递推公式:题⽬已经把递推公式直接给我们了:状态转移⽅程 dp[i] = dp[i - 1] + dp[i - 2]
  3. 初始化dp数组: dp[0] = 0, dp[1] = 1
  4. 确定遍历顺序: dp[i] 是依赖 dp[i - 1] 和 dp[i - 2] ,所以遍历的顺序是从前到后遍历
int fib(int n){
    if(n<2)  return n;
	int dp[n+1];
	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)
根据斐波那契数列的性质,我们可以用 滚动数组来优化一下空间复杂度

int fib(int n){
    if(n<2)  return n;
	int dp[2]={0,1};
	int result=0;
	for(int i=2;i<=n;++i){
		result=dp[0]+dp[1];
		dp[0]=dp[1];
		dp[1]=result;
	}
	return dp[1]; 
}

时间复杂度:O(n)
空间复杂度:O(1)

方法四:快速矩阵幂

定义一个矩阵:
在这里插入图片描述

可以找到一个矩阵M(这里可以理解为从一个状态转移到另一个状态),使得:

在这里插入图片描述

设:
在这里插入图片描述

所以
在这里插入图片描述

由矩阵乘法可以得到:

af(n)+bf(n-1)=f(n+1)
cf(n)+df(n-1)=f(n)
最终求得:
在这里插入图片描述

由递推公式最终得到:
在这里插入图片描述

因此只要我们能快速计算矩阵 M 的 k 次幂,就可以得到 F(n)的值。
此处需要用到快速幂的算法,不明白的可以看一下快速幂算法_Sakeyuan的博客-CSDN博客:

//矩阵相乘 
vector<vector<int>> array_multiply(vector<vector<int>>& a, vector<vector<int>>& b){  
    vector<vector<int>> c{ {0, 0}, {0, 0} };
    for (int i = 0; i < 2; i++) {
        for (int j = 0; j < 2; j++) {
            c[i][j] = a[i][0] * b[0][j] + a[i][1] * b[1][j];
        }
    }
    return c;
}
vector<vector<int>> array_pow(vector<vector<int>>& a, int n) {   //矩阵快速幂 
    vector<vector<int>> ret{ {1, 0}, {0, 1} };     //单位矩阵 ,储存结果 
    while (n > 0) {
        if (n & 1) {
            ret = array_multiply(ret, a);
        }
        n >>= 1;
        a = array_multiply(a, a);
    }
    return ret;
}
 
int fib(int n) {
    if (n < 2) {
        return n;
    }
    vector<vector<int>> m{ {1, 1}, {1, 0} };
    vector<vector<int>> result = array_pow(m, n);
    return result[0][1];       // 由递推公式可以看出f(n)为 result[0][1]位置 
} 

时间复杂度:O(logn)
空间复杂度:O(1)

五、通项公式:

根据递推公式 F(n)=F(n - 1)+F(n - 2) 可以得出特征方程:

在这里插入图片描述

求得:
在这里插入图片描述

通解为:
在这里插入图片描述

带入f(0)=0,f(1)=1,解得:
在这里插入图片描述

最终得到通项公式:
在这里插入图片描述

int fib(int n) {
    double sqrt_5 = sqrt(5);
    double fib_n = pow((1 + sqrt_5) / 2, n) - pow((1 - sqrt_5) / 2, n);
    return round(fib_n / sqrt_5);  /*round()函数四舍五入*/
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

决策树-剪枝处理

前言&#xff1a;理解《机器学习》P79-83中的决策树剪枝示例。 决策树生成 原始数据集如下所示&#xff0c;前10行为训练集&#xff0c;后7行为验证集&#xff0c;由此数据集可生成如下所示的决策树。 下面解释未进行剪枝操作的决策树为何如上图所示。 不对解释每个结点和分支…

WPF-3D图形

WPF-3D图形 WPF的3D功能可以在不编写任何c#代码的情况下进行绘制&#xff0c;只需要使用xaml即可完成3D图形的渲染。本文主要讲述了WPF-3D中的关键概念&#xff0c; 以及常用到的命中测试、2d控件如何在3D对象中进行渲染&#xff0c;除此之外&#xff0c;还演示了如何导入外部…

InstanceNorm LayerNorm

InstanceNorm && LayerNorm author: SUFEHeisenberg date: 2023/01/26 先说结论: 将Transformer类比于RNN&#xff1a;一个token就是一层layer&#xff0c;对一整句不如token有意义原生Bert代码或huggingface中用的都是InstanceNorm instead of LayerNorm&#xff…

【AAAI2023】Head-Free Lightweight Semantic Segmentation with Linear Transformer

论文&#xff1a;【AAAI2023】Head-Free Lightweight Semantic Segmentation with Linear Transformer 代码&#xff1a;https://github.com/dongbo811/AFFormer 这是来自阿里巴巴的工作&#xff0c;作者构建了一个轻量级的Transformer网络用于语义分割&#xff0c;主要有两点…

发现下属的学历造假,但是他的工作能力又很强,该开除他吗?

在职场上混&#xff0c;学历是敲门砖还是定音锤呢&#xff1f;一位网友问&#xff1a;发现下属的学历造假&#xff0c;但是他的工作能力又很强&#xff0c;该开除他吗?有人觉得一定要开除&#xff0c;这就是钻空子&#xff0c;受影响最大的人不是他&#xff0c;而是那些真才实…

上采样与下采样

数据分析中的上采样和下采样 背景&#xff1a; 在分类问题中&#xff0c;由于各种原因&#xff0c;我们所获取到的数据集很容易出现正负样本的不平衡&#xff0c;或者某些数据特别多&#xff0c;有些数据则特别少&#xff0c;在这样的数据集中&#xff0c;进行训练&#xff0c…

OpenCV直方图Java 演示程序

直方图Java 演示程序以下文件编码为utf-8 为佳。代码文件名&#xff1a;OpenCvMain.javapackage org.opencv;import java.net.URL;import java.util.LinkedList;import java.util.List;import org.opencv.core.Core;import org.opencv.core.CvType;import org.opencv.core.Mat;…

Linux常用命令——setpci命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) setpci 查询和配置PCI设备的使用工具 补充说明 setpci命令是一个查询和配置PCI设备的使用工具。 语法 setpci(选项)(参数)选项 -v&#xff1a;显示指令执行的细节信息&#xff1b; -f&#xff1a;当没有任何…

Opencv形态学操作——腐蚀、膨胀、梯度、开运算、闭运算、礼帽、黑帽(附案例详细讲解及可执行代码)

Opencv形态学操作 腐蚀膨胀梯度开运算闭运算礼帽黑帽总结腐蚀 在地理或者化学中,我们学习过腐蚀,是指在某种 作用下产生损耗与破坏的过程。你也可以理解为减肥。在Opencv中,腐蚀操作可以使白色轮廓变小,也就是说可以去除一些白色的噪声。 如果你接触过卷积核的话,腐蚀就更…

【JavaSE专栏8】运算符、表达式和语句

作者主页&#xff1a;Designer 小郑 作者简介&#xff1a;Java全栈软件工程师一枚&#xff0c;来自浙江宁波&#xff0c;负责开发管理公司OA项目&#xff0c;专注软件前后端开发&#xff08;Vue、SpringBoot和微信小程序&#xff09;、系统定制、远程技术指导。CSDN学院、蓝桥云…

盖子的c++小课堂——第十三讲:二维数组

前言 过了几天了&#xff0c;终于有时间更新了&#xff0c;有个通知&#xff0c;以后我不用颜色区分了&#xff0c;不然换了背景看不见&#xff0c;理解一下&#xff0c;蟹蟹~~ 举例 作者&#xff1a;一下是某次奥运会的奖牌榜&#xff0c;你知道如何储存奖牌榜吗~~ 粉丝&am…

机器学习中软投票和硬投票的不同含义和理解

设置一个场景&#xff0c;比如对于今天音乐会韩红会出现的概率三个人三个观点 A&#xff1a;韩红出现的概率为47% B&#xff1a;韩红出现的概率为57% C&#xff1a;韩红出现的概率为97% 软投票&#xff1a;软投票会认为韩红出现的概率为1/3*(47%57%97%)67% 硬投票&#xff1a;…

“子序列问题”系列总结,一文读懂(Java实现)

目录 前言 一、最长递增子序列 1.1、dp定义 1.2、递推公式 1.3、初始化 1.4、注意 1.5、解题代码 二、最长连续递增序列 2.1、分析 2.2、解题代码 三、最长重复子数组 3.1、dp定义 3.2、递推公式 3.3、初始化 3.4、解题代码 四、最长公共子序列 4.1、分析 4.2…

Opencv项目实战:20 单手识别数字0到5

目录 0、项目介绍 1、效果展示 2、项目搭建 3、项目代码展示 HandTrackingModule.py Figures_counter.py 4、项目资源 5、项目总结 0、项目介绍 今天要做的是单手识别数字0到5&#xff0c;通过在窗口展示&#xff0c;实时的展示相应的图片以及文字。 在网上找了很久的…

硬核来袭!!!一篇文章教你入门Python爬虫网页解析神器——BeautifulSoup详细讲解

文章目录一、BeautifulSoup介绍二、安装三、bs4数据解析的原理四、bs4 常用的方法和属性1、BeautifulSoup构建1.1 通过字符串构建1.2 从文件加载2、BeautifulSoup四种对象2.1 Tag对象2.2 NavigableString对象2.3 BeautifulSoup对象2.4 Comment对象五、contents、children与desc…

springboot自定义拦截器的简单使用和一个小例子

springboot自定义拦截器的使用1. 自定义拦截器2. 拦截器登录验证的小demo2.1 配置pom.xml2.2 创建User的bean组件2.3 创建需要的表单页面以及登录成功的页面2.4 编写controller映射关系2.5 自定义拦截器类&#xff0c;实现intercepetor接口2.6注册添加拦截器&#xff0c;自定义…

【SpringCloud】Nacos集群搭建

集群结构图官方给出的Nacos集群图如下&#xff1a;其中包含3个nacos节点&#xff0c;然后一个负载均衡器代理3个Nacos。这里负载均衡器可以使用nginx。我们接下来要尝试 Nacos集群搭建&#xff0c;效果图如下所示&#xff1a;三个nacos节点的地址&#xff1a;节点ipportnacos1l…

二、Java框架之Spring注解开发

文章目录1. IOC/DI注解开发1.1 Component注解ComponentController Service Repository1.2 纯注解开发模式1.3 注解开发bean管理ScopePostConstruct PreDestroy1.4 注解开发依赖注入Autowired QualifierValuePropertySource1.5 第三方bean管理Beanimport&#xff08;多个Config类…

Redisson 完成分布式锁

1、简介 Redisson 是架设在 Redis 基础上的一个 Java 驻内存数据网格&#xff08;In-Memory Data Grid&#xff09;。充分 的利用了 Redis 键值数据库提供的一系列优势&#xff0c;基于 Java 实用工具包中常用接口&#xff0c;为使用者 提供了一系列具有分布式特性的常用工具类…

JavaWeb | 揭开SQL注入问题的神秘面纱

本专栏主要是记录学习完JavaSE后学习JavaWeb部分的一些知识点总结以及遇到的一些问题等&#xff0c;如果刚开始学习Java的小伙伴可以点击下方连接查看专栏 本专栏地址&#xff1a;&#x1f525;JDBC Java入门篇&#xff1a; &#x1f525;Java基础学习篇 Java进阶学习篇&#x…