百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

一文解读斐波那契数列原理及生成器实现

off999 2024-10-23 12:40 18 浏览 0 评论

斐波那契数列

斐波那契数列(意大利语:Successione di Fibonacci),又译为菲波拿契数列、菲波那西数列、斐氏数列、黄金分割数列。

在数学上,斐波那契数列是以递归的方法来定义:


用文字来说,就是斐波那契数列由0和1开始,之后的斐波那契数就是由之前的两数相加而得出。首几个斐波那契数是:

0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 233……(OEIS中的数列A000045)

特别指出:0 不是第一项,而是第零项,第一项是第一项!!

以斐波那契数为边的正方形拼成的近似的黄金矩形(1:1.618)


以斐波那契数为半径的圆拼成的近似的黄金螺旋螺旋(1:1.618)


斐波那契数列之源起

2.1 兔子学说

公元1150年印度数学家Gopala和金月在研究箱子包装对象长宽刚好为1和2的可行方法数目时,首先描述这个数列。在西方,最先研究这个数列的人是比萨的列奥那多(意大利人斐波那契Leonardo Fibonacci),他描述兔子生长的数目时用上了这数列:

第一个月初有一对刚诞生(先得长大两月,第三月才能生)的兔子

第二个月之后(第三个月初)它们可以生育

每月每对可生育的兔子会诞生下一对新兔子

兔子永不死去

如下图,兔子对 的数量 就是斐波那契数列。值得注意的是,假设兔子是永不会死去的,其中带红蓝点的兔子代表可以生小兔,反之不能。(兔兔在诞生的第三个月才能生小兔)


假设在n月有兔子总共a对,n+1月总共有b对。在n+2月必定总共有a+b对:因为在n+2月的时候,前一月(n+1月)的b对兔子可以存留至第n+2月(在当月属于新诞生的兔子尚不能生育)。而新生育出的兔子对数等于所有在n月就已存在的a对

如下图,斐波纳契数是帕斯卡三角形的每一条红色对角线上数字的和


2.2 数列表达式

为求得斐波那契数列的一般表达式,可以借助线性代数的方法。高中的初等数学知识也能求出。


斐波那契数列之编程

需求:用三种方法实现斐波那契数列的第1-10项,注意这里不包括第0项。

3.1 方法一:循环实现 Loop

coding:

def fibonacci_loop(n):
    first, second, count, fib = 0, 1, 2, []
    if n == 1:
        fib.append(second)
    else:
        fib.append(second)
        while count <= n:
            tmp = second
            second = first + second
            first = tmp
            fib.append(second)
            # 更新值
            count += 1
    return fib


res_fib = fibonacci_loop(10)
print(res_fib)

输出:

[1, 1, 2, 3, 5, 8, 13, 21, 34, 55]

3.3 方法三:递归实现 Recursion

coding:

def fibonacci_recursion(n):
    if n == 1 or n == 2:
        return 1
    else:
        return fibonacci_recursion(n-1) + fibonacci_recursion(n-2)


res = [fibonacci_recursion(n) for n in range(1, 11)]
print(res)

输出:

[1, 1, 2, 3, 5, 8, 13, 21, 34, 55]

3.4 上述三种方法耗时PK

需求:分别用循环、生成器、递归来实现第1到第40项的斐波那契数列,对比耗时情况,猜猜看哪个耗时最久?

from functools import wraps
import time


# 计时装饰器
def timmer(func):
    @wraps(func)
    def inner(*args, **kwargs):
        start_time = time.time()
        res = func(*args, **kwargs)
        end_time = time.time()
        print(f'本次{func.__name__}函数执行共耗时{int(end_time-start_time)}s')
        return res
    return inner


@timmer
def fibonacci_loop(n):
    first, second, count, fib = 0, 1, 2, []
    if n == 1:
        fib.append(second)
    else:
        fib.append(second)
        while count <= n:
            tmp = second
            second = first + second
            first = tmp
            fib.append(second)
            # 更新值
            count += 1
    return fib


@timmer
def fibonacci_genarator(n):
    n1, n2, count = 0, 1, 2
    yield n2
    while count <= n:
        tmp = n2
        n2 = n1 + tmp
        n1 = tmp
        yield n2
        count += 1


def fibonacci_recursion(n):
    if n == 1 or n == 2:
        return 1
    else:
        return fibonacci_recursion(n-1) + fibonacci_recursion(n-2)


if __name__ == '__main__':
    # 循环
    res_fib = fibonacci_loop(40)
    print(res_fib)
    # 生成器
    res = fibonacci_genarator(40)
    print(list(res))
    # 递归
    start_time = time.time()
    res = [fibonacci_recursion(n) for n in range(1, 41)]
    print(res)
    end_time = time.time()
    print(f'本次{fibonacci_recursion.__name__}函数执行共耗时{int(end_time-start_time)}s')

如下,递归耗时最久,竟然耗时50秒,循环和生成器耗时不到1秒,所以说递归需慎用~



最后多说一句,小编是一名python开发工程师,这里有我自己整理了一套最新的python系统学习教程,包括从基础的python脚本到web开发、爬虫、数据分析、数据可视化、机器学习等。想要这些资料的可以关注小编,并在后台私信小编:“01”即可领取。

相关推荐

「Python条件结构」if…else实现判断奇偶数

功能要求用户从键盘上输入一个整数,判断该数是奇数还是偶数。说明:能被2整除的整数叫偶数,不能被2整除的叫奇数;即该数除以2后余数为0时该数为偶数,否则该数为奇数。求余数运算符为“%”。实例代码num...

Python if else条件语句详解

前面我们看到的代码都是顺序执行的,也就是先执行第1条语句,然后是第2条、第3条……一直到最后一条语句,这称为顺序结构。但是对于很多情况,顺序结构的代码是远远不够的,比如一个程序限制了只能成年人使用,儿...

python基础篇: python中的流程控制,你都了解吗?

在之前的文章中大致的介绍过python中的流程控制语句,今天通过一些案例来详细了解一下python中的流程语句。目前python中流程控制语句,包含如下,如有遗漏欢迎留言补充。在python中条件判断...

python中if语句

if语句用来判断,当不同的条件成立去做与之对应事情;格式如下:if条件:执行代码条件为True才会去做执行代码布尔类型(bool)说到布尔类型,就像开关只有两个值一样,布尔类型的值只有两个...

python中的循环语句到底难不难

好多初学者会有一种这样的心里:循环难不难?该怎么学习?下面来给大家分析下.Python中的循环语句并不难,但需要理解其核心逻辑和应用场景。以下是针对零基础学习者的清晰解析,通过对比、示例和常见误...

Python6大基础运算符,看完这篇之后会让你有一个彻底认识

昨天我们准备好了Python程序所需要的的东西,那么今天我们开始了解Python的各种基础运算符,这些要是不熟悉下来你后面的路也会走的很艰难Python支持基础运算符,常见的算术运算符有+、-、*、/...

Python基础:条件语句和循环语句

下面会详细讲解一下Python关于条件语句和循环语句,会包含一些示例代码。我们首先来介绍条件语句(if-else),然后再讨论循环语句(for和while循环)。条件语句(if-else)在Pytho...

Python合集之Python循环语句(一)

在上一节的合集中,我们了解了Python流程控制语句中if语句的嵌套及条件表达会的相关知识,本节我们将进一步了解一下Python循环语句中的while语句的相关知识。在日常生活中很多问题都无...

Python“三步”即可爬取,毋庸置疑

声明:本实例仅供学习,切忌遵守robots协议,请不要使用多线程等方式频繁访问网站。#第一步导入模块importreimportrequests#第二步获取你想爬取的网页地址,发送请求,获取网页内...

「Python条件结构」if…else实现三角形判断

功能要求编写程序,判断输入的三个数是否能构成三角形的三个边。如果可以,打印“可以构成三角形”;如果不可以,打印“不可以构成三角形”。构成三角形的条件是:三条边都等于0,且任何2条边的边长之和都大于第三...

Python中检查对象是否具有某个属性的方法

技术背景在Python编程中,经常会遇到需要检查一个对象是否具有某个特定属性的情况。例如,在调用对象的属性之前,需要先确认该属性是否存在,以避免引发AttributeError异常。以下将介绍几种常见...

Python条件语句:从入门到精通

导语条件语句是编程中的基础概念,它允许我们根据不同的条件执行不同的代码块。在Python中,条件语句的灵活性和易读性使其成为编写逻辑判断和流程控制的强大工具。本教程将带您深入了解Python条件语句的...

简单学Python——条件语句if

条件语句是用来判断给定的条件是否满足(表达式值是否为0或False),并根据判断的结果(真或假)决定执行的语句。Python条件语句用的是if或if和else、elif等搭配实现的。代码执行的过程:i...

Python合集之Python跳转语句(一)

在上一节的合集中,我们了解了Python循环嵌套语句的相关知识,本节我们将进一步了解一下Python跳转语句中的break的相关知识。当循环条件一直满足时,程序会一直执行下去,如果希望在中间离开循环...

新手学Python避坑,学习效率狂飙! 八、Python 布尔值判断

布尔值判断系统知识在Python里,布尔类型仅有两个值:True和False,它们常被用于条件判断。下面从几个方面展开介绍:1.布尔运算逻辑与(and):只有当两个操作数都为True时,...

取消回复欢迎 发表评论: