『C#』如何用递归计算斐波那契数列的第 100000 项 ?

2024-01-23 01:40

本文主要是介绍『C#』如何用递归计算斐波那契数列的第 100000 项 ?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文最后更新于 2019年 5月 6号 下午 4点 16分,并同步发布于 :

  • 简书 —— 创作你的创作
  • CSDN —— 专业 IT 技术社区
  • www.tobinary.art —— 我的博客

声明 : 请不要使用本文的代码直接用于实际项目,本文的目的是以这个示例给读者提供一点编程上的思路

本文假设读者有对如下概念有所了解 :

  • 委托
  • Lambda 表达式
  • 函数柯里化
  • 本地函数

如何用递归计算斐波那契数列的第 100000 项 ?
有的同学可能会说 : 那还不简单, 不到一分钟便写出了如下代码 :

public static int Fibonacci(int n)
{if (n == 1 || n == 2){return 1;}return Fibonacci(n - 1) + Fibonacci(n - 2);
}

或者 :

public static int Fibonacci(int n)=> n <= 2? 1: Fibonacci(n - 1) + Fibonacci(n - 2);

先测试一下斐波那契数列的第 10 项试试 :


再试试第 50 项 :

结果竟然是负数 ? 不过仔细一想, 是因为结果值超出了 int 类型的存储范围。而且用了 一分多钟 的时间 !

100 项呢 :
emmmmmmm… … 由于我生命有限, 就不测试这个了, 估计我睡完一觉了还没得出结果 orz

如果计算更大的项时, 很可能出现下面的情况 : (具体多大的项 因电脑配置而异)

程序计算比较大的项时, 发生了 堆栈溢出 异常。

递归计算斐波那契数列时, 函数调用的次数是 指数级 增加的


在解决问题之前我们先看一下递归求斐波那契数列的函数调用情况 :

可以看到, 在计算结果时, 出现了很多重复计算
这也是为什么计算比较大的项时, 用时非常长

将值缓存起来

那么应该如何改写上面写的函数呢 ?
我们可以通过把已经计算过的值存储在一个容器中, 等下次需要计算时, 直接从中取值。
这样可以大幅度的减少函数递归调用的次数

当然还有一个问题 :
计算比较大的项时, 值可能非常大, 超出了所有C#内置的基本整数类型的最大存储范围。
所以我们需要使用 System.Numerics 命名空间下的大整数类型 : BigInteger, 这个类型可以存储任意大的整数。当然也可以自己手写一个大整数类型。至于怎么写一个大整数类型,不在本文的讨论范围。

.NET framework 程序需要添加程序集引用: System.Numerics.dll


现在开始修改刚刚写的递归函数

这样的话, 就避免了大量的重复计算。

先试试计算斐波那契数列的第 50 项 :

仅仅用了 10 毫秒 !

再试试第 5000 项呢 ?

仅仅用了 19 毫秒 ! 第 5000 项的值已经非常非常大了

虽然使用缓存避免了大量的重复计算,使得计算时间大幅降低
但是需要计算的项非常大时, 调用栈还是会发生溢出 !

渐进式计算 :

前面我们通过把中间结果缓存到集合中,以避免重复计算,也就是说 :
如果前 5000 项已经计算过,那么再计算第 5001 项时,只需一次计算 (第 5000 项和第 4999 项相加)

现在假设计算第 6000 项时会发生 栈溢出,那么我们可以 :

  1. 先计算第 5000 项 ( 不会 栈溢出)
  2. 再计算第 6000 项 ( 也不会 发生栈溢出 !)

因为前 5000 项已经被缓存到集合中,所以再计算第 6000 项时,只需计算第 5001 ~ 6000 项。

那么我们需要计算第 100000 (十万) 项呢 ?

  1. 计算第 5000
  2. 计算第 10000
  3. 计算第 15000
  4. 计算第 20000
    … …

以此类推
最后计算第 100000

现在我们修改上面的代码,使得能计算 任意大 的项 (只要运行内存够大):

.NET Framework 4.0 及以下版本,单个对象不能大于 2GB

使用上图中的代码计算斐波那契数列的第 100000 (十万)项 :


这是一个 20900 位的整数,用时 0.9


如果想达到这样的效果, 又不想在外面定义一个类成员怎么办?

  • 可以把缓存结果值的容器直接放在函数中,

    但是每次调用函数时, 都会创建一个集合对象, 开销会比较大。

  • 使用 闭包

使用闭包延长局部变量的生命周期 :

Fibonacci 函数执行结束时, cache 这个局部变量仍可以从函数外部访问,不会被 GC 释放

如何使用 ?

这三次函数调用不会创建集合对象

如果只需要调用一次则可以直接调用返回的函数 :


以上代码没有经过仔细优化,目的是给读者提供一点思路

源代码 : 点击这里获取源代码

—END—

这篇关于『C#』如何用递归计算斐波那契数列的第 100000 项 ?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/634868

相关文章

C#中lock关键字的使用小结

《C#中lock关键字的使用小结》在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时,其他线程无法访问同一实例的该代码块,下面就来介绍一下lock关键字的使用... 目录使用方式工作原理注意事项示例代码为什么不能lock值类型在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时

C# $字符串插值的使用

《C#$字符串插值的使用》本文介绍了C#中的字符串插值功能,详细介绍了使用$符号的实现方式,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录$ 字符使用方式创建内插字符串包含不同的数据类型控制内插表达式的格式控制内插表达式的对齐方式内插表达式中使用转义序列内插表达式中使用

C#中的Converter的具体应用

《C#中的Converter的具体应用》C#中的Converter提供了一种灵活的类型转换机制,本文详细介绍了Converter的基本概念、使用场景,具有一定的参考价值,感兴趣的可以了解一下... 目录Converter的基本概念1. Converter委托2. 使用场景布尔型转换示例示例1:简单的字符串到

C#监听txt文档获取新数据方式

《C#监听txt文档获取新数据方式》文章介绍通过监听txt文件获取最新数据,并实现开机自启动、禁用窗口关闭按钮、阻止Ctrl+C中断及防止程序退出等功能,代码整合于主函数中,供参考学习... 目录前言一、监听txt文档增加数据二、其他功能1. 设置开机自启动2. 禁止控制台窗口关闭按钮3. 阻止Ctrl +

C#解析JSON数据全攻略指南

《C#解析JSON数据全攻略指南》这篇文章主要为大家详细介绍了使用C#解析JSON数据全攻略指南,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、为什么jsON是C#开发必修课?二、四步搞定网络JSON数据1. 获取数据 - HttpClient最佳实践2. 动态解析 - 快速

C#连接SQL server数据库命令的基本步骤

《C#连接SQLserver数据库命令的基本步骤》文章讲解了连接SQLServer数据库的步骤,包括引入命名空间、构建连接字符串、使用SqlConnection和SqlCommand执行SQL操作,... 目录建议配合使用:如何下载和安装SQL server数据库-CSDN博客1. 引入必要的命名空间2.

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

C#中Guid类使用小结

《C#中Guid类使用小结》本文主要介绍了C#中Guid类用于生成和操作128位的唯一标识符,用于数据库主键及分布式系统,支持通过NewGuid、Parse等方法生成,感兴趣的可以了解一下... 目录前言一、什么是 Guid二、生成 Guid1. 使用 Guid.NewGuid() 方法2. 从字符串创建

C# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字