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

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


斐波那契数列

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

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


如何用递归计算斐波那契数列的第 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


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


修改之后的 Fibonacci 函数

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

先试试计算斐波那契数列的第 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

修改之后的 Fibonacci 函数

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

斐波那契数列的第十万项的值

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


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

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

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

  • 使用 闭包

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

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

返回一个函数的函数

如何使用 ?

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

调用

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


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

---END---

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351