Fibonacci,懂你(2011-04-03)

請用「Mou」打開😅


豆瓣上看到有人推荐 S. Dasgupta, C. H. Papadimitriou 和 U. V. Vazirani 写的Algorithms,随手翻了下⋯⋯

学计算机的没有不知道 Hanoi塔和 Fibonacci 数列的。但 Hanoi 塔问题是死的,而求解 Fibonacci 数列却精彩不断。

Fibonacci 数列$$$F_0, F_1, F_2, ...$$$是这样定义的:

$$ F_0 = 0, F_1= 1, F_n = F_{n-1} + F_{n-2} $$

如果用定义来求解,计算$$$F_n$$$至少需要$$$F_{n-1} + F_{n-2} + 1$$$次加法(还不算对递归终点的判断),也就是说用这种策略计算$$$F_n$$$所需的加操作不少于其自身。那$$$F_n$$$是个什么数量级呢?$$$F_n ≥ 2^{0.5n} (n ≥ 6)$$$

不信?

$$F_0 = 0, F_1 = 1, F_2 = 1, F_3 = 2, F_4 = 3, F_5 = 5, F_6 = 8, ...$$

$$2^{0/2} = 1, 2^{1/2} = \sqrt{2}, 2^{2/2} = 2, 2^{3/2} = 2\sqrt{2}, 2^{4/2} = 4, 2^{5/2} = 4\sqrt{2}, 2^{6/2} = 8, ...$$

现在开始绑定自然数(数学归纳啦):

设$$$F_k ≥ 2^{k/2} (6 ≤ k < n)$$$,则

$$F_n = F_{n-1} + F_{n-2} ≥ 2·F_{n-2} ≥ 2·2^{(n-2)/2} = 2^{n/2}$$

0.5 应该是个比较糙的估计,上限是多少呢?

设$$$F_k ≥ 2^{ck} (6 ≤ k < n)$$$,则

$$F_n = F_{n-1} + F_{n-2} ≥ 2^{c(n-1)} + 2^{c(n-2)} = 2^{c(n-2)} (2^c + 1) ≥ 2^{cn}$$

即$$$2^c + 1 ≥ 2^{2c}$$$

简单吧,一元二次方程。令$$$t = 2c$$$,则$$$t2 - t -1 ≤ 0$$$。还因为{$$$F_n$$$}递减,所以

\[2^c > 1 = 20, \
0 < t ≤ {{1+\sqrt{5}} \over 2}, \
c ≤ log_2(1+\sqrt{5}) - 1 ≈ 0.694\]

把不等号方向颠倒还可以得到$$$F_n ≤ 2^{0.694n}$$$,故$$$F_n = Θ(2^{0.694n})$$$。

故事并没有结束,注意到算$$$F_n$$$需要知道$$$F_{n-1}$$$和$$$F_{n-2}$$$,而$$$F_{n-2}$$$在算$$$F_{n-1}$$$时已经算过一次。如果我们把计算结果都存上(memorization),完全可以在线性时间内求解。更进一步,存储整个数列是否必要,我们需要的只不过是最后两项啊。由此可以得到自底向上版本(动态规划?):

def fib(n)  
  fn_1, fn_2 = 0, 1  
  (1..n).each {  
    fn_1, fn_2 = (fn_1 + fn_2), fn_1  
  }  
  fn_1  
end  
p fib 7 # >> 13  

都$$$O(n)$$$了,但故事还没有结束。你可能知道 Fibonacci 数列有个解析解(就是一个直接计算,不用递归的公式):

$$$\sqrt{5}$$$是个无理数,但结果确是整数😱。用浮点肯定会有误差,而且还不知道会不会发散⋯⋯再说就算用符号计算还不是得乘n次,复杂度没降啊。此路不通!
但是,故事还是没有结束。

虽然只是换了种写法,但世界因此不同o。谁说幂运算是线性的!看好了,$$$2^4 = (22)2 = 4^2 = 16$$$可只用了两次(准确的说是⌈log2 4 ⌉次)乘法!

def matrix2_2_times(a, b)  
  [[a[0][0]*b[1][0] + a[0][1]*b[1][0],  
    a[0][0]*b[0][1] + a[0][1]*b[1][1]],  
   [a[1][0]*b[0][0] + a[1][1]*b[1][0],  
    a[1][0]*b[0][1] + a[1][1]*b[1][1]]]  
end  
def matrix_power(a, n)  
  return a if n == 1  
  m = matrix_power(a, n/2)  
  if n % 2 == 0  
    pow = matrix2_2_times(m, m)  
  else  
    pow = matrix2_2_times(a,  
            matrix2_2_times(m, m))  
  end  
  pow  
end  
u = [[0, 1], [1, 1]]  
matrix_power(u, 6) # => [[6, 8], [8, 13]]  

复杂度是$$$O(log n)$$$!

故事该结束了吧⋯⋯还没有。

是,我们只用了$$$O(log n)$$$次加法和乘法,但当数项不断增大,加法和乘法都不能在在常数时间内完成。时间复杂度的下限到底是多少?这个正在看😓

P.S. 学算法最痛快就是看着别人一点点改进(比如《编程珠玑》里提到的对N体问题的改进),就像看 Harry Potter 一点点长大;

P.P.S 学算法最痛苦就是学了却不知道别人怎么想出来的⋯⋯

P.P.P.S 故意把小 Ron 弄掉的,邪恶吧o

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

推荐阅读更多精彩内容

  • 计算机二级C语言上机题库(南开版) 1.m个人的成绩存放在score数组中,请编写函数fun,它的功能是:将低于平...
    MrSunbeam阅读 6,336评论 1 42
  • 一生 多少次选择 多少次惊喜 多少次失望 多少次跌倒 …… 不用管 有多少次 试问 是你自己的选择吗……
    CHENXI_G阅读 94评论 0 0
  • 记得一句话,好像是某个作家名句,大致是人生最可悲的是与自己感到孤独的人共度余生。说实在,这句话也曾让深夜里的我感到...
    SUE用生命起舞阅读 201评论 0 0
  • 我爱她,每一段陪我走过跌跌撞撞的年华; 我爱她,每一个沮丧孤独唯有她在的夜晚; 我爱她,每一句不依不饶不让我犯错的...
    叹叹叹息阅读 90评论 0 0
  • CABasicAnimation *animation=[CABasicAnimation animationWi...
    法库德阅读 214评论 0 0