遞歸是個有意思的概念,正如在前面所說,遞歸能讓算法的可讀性大大提高,而且通常要比使用循環(huán)結(jié)構更能寫出準確的算法。這本書形象引入了遞歸,并沒有太深入,所以我進行了一點“添油加醋”。
遞歸 概念遞歸其實就是自己調(diào)用自己??梢詮亩喾N維度對遞歸分類,我見過的最常見的分類:
直接遞歸
自己直接調(diào)用自己。如:
--haskell length" :: [a] -> Int length" [] = 0 length" (_:xs) = 1 + length" xs
上面定義的length"就是通過直接遞調(diào)用自身完成列表長度的計算。
間接遞歸
可以認為只要不是直接調(diào)用自己的遞歸都是間接遞歸,其表現(xiàn)形式較多,如A->(調(diào)用)B,B->(調(diào)用)A,如奇偶謂詞函數(shù):
--haskell odd" :: Int -> Bool odd" 0 = False odd" n = even" (n - 1) even" :: Int -> Bool even" 0 = True even" n = odd" (n - 1)
也可以有A->B,B->C,... Z->A。這里就不舉例子了。
書中的例子有一個盒子,盒子里套著一個或多個盒子,盒子里的盒子又有盒子,依次類推。而鑰匙就在某個盒子里,我們怎么找到鑰匙呢。策略一
偽代碼如下:[偽代碼是對手頭問題的簡要描述,看著像代碼,但其實更接近自然語言。]
def look_for_key(main_box): pile = main_box.make_a_pile_to_look_through() while pile is not empty: box = pile.grab_a_box() for item in box: if item.is_a_box(): pile.append(item) elif item.is_a_key(): print "found the key!"
如果學習過樹的遍歷,是不是發(fā)現(xiàn)這就是廣度優(yōu)先遍歷啊
策略二
偽代碼如下:
def look_for_key(box): for item in box: if item.is_a_box(): look_for_key(item) elif item.is_a_key(): print "found the key!"
對應的就是深度優(yōu)先遍歷啊
這兩種方法的作用相同,但第二種方法更清晰。遞歸只是讓解決方案更清晰,并沒有性能上的優(yōu)勢。實際上,在有些情況下,使用循環(huán)的性能更好。Leigh Caldwell在Stack Overflow上說的一句話:“如果使用循環(huán),程序的性能可能更高;如果使用遞歸,程序可能更容易理解。如何選擇要看什么對你來說更重要?!?/p> 基線條件和遞歸條件
每個遞歸函數(shù)都有兩部分:基線條件(base case)和遞歸條件(recursive case)。遞歸條件指的是函數(shù)調(diào)用自己,而基線條件則指的是函數(shù)不再調(diào)用自己,從而避免形成無限循環(huán)。遞歸條件一定是向基線條件靠攏的,否則,只能無限遞歸下去。如
#python def countdown(i): print i if i <= 0: return else: countdown(i-1)
上述函數(shù)中i的值收斂于0,即達到基線條件,從而不會無限遞歸下去。
棧主要講了與遞歸相關的一種數(shù)據(jù)結(jié)構--棧(stack)。棧是一種支持FILO(First In Last Out)的數(shù)據(jù)結(jié)構。
為什么要提這種數(shù)據(jù)結(jié)構呢?
這是因為現(xiàn)代計算機對函數(shù)的實現(xiàn)用到了調(diào)用棧(call stack)的棧。調(diào)用函數(shù)時,計算機將首先為該函數(shù)調(diào)用分配一塊內(nèi)存。每當你調(diào)用函數(shù)時,計算機都將函數(shù)調(diào)用涉及的所有變量的值存儲到內(nèi)存中。
假設A函數(shù)調(diào)用B函數(shù),此時計算機為兩個函數(shù)分配了內(nèi)存,暫且稱之為A函數(shù)內(nèi)存和B函數(shù)內(nèi)存,它們的位置關系如下:
----棧頂----
B函數(shù)內(nèi)存
—————
A函數(shù)內(nèi)存
—————
若B函數(shù)執(zhí)行完,計算機就可以回收B函數(shù)內(nèi)存了,即從棧頂彈出B函數(shù)內(nèi)存,此時只有A函數(shù)內(nèi)存了。
----棧頂----
A函數(shù)內(nèi)存
—————
以上操作符合FILO的定義,調(diào)用棧是棧的一種具體應用。
那如果調(diào)用棧數(shù)量太多,會有什么后果呢?
遞歸調(diào)用棧#python def fact(x): if x == 1: return 1 else: return x * fact(x-1)
對于較小的正整數(shù),這個程序沒有問題;而如果x較大,在fact執(zhí)行的時會發(fā)現(xiàn)內(nèi)存量會飆升,甚至會出現(xiàn)程序無法正常執(zhí)行下去。這是因為此時遞歸調(diào)用棧的情況類似:
----棧頂----
fact(1)函數(shù)內(nèi)存
———————
...
———————
fact(n-2)函數(shù)內(nèi)存
———————
fact(n-1)函數(shù)內(nèi)存
———————
fact(n)函數(shù)內(nèi)存
———————
fact(n)依賴fact(n-1),依次類推,導致計算機存儲了大量函數(shù)調(diào)用的信息。這類問題大體有兩種解決方式:
重新編寫代碼,轉(zhuǎn)而使用循環(huán)。
使用尾遞歸?,F(xiàn)在已經(jīng)有很多編譯器支持尾遞歸優(yōu)化了。
我的公眾號地址
文章版權歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/43432.html
摘要:再談大表示法快速排序的獨特之處在于其速度取決于選擇的基準值。在平均情況下快速排序的運行時間為在最糟情況下退化為??焖倥判蚝秃喜⑴判虻乃惴ㄋ俣确謩e表示為和,是算法所需的固定時間量被稱為常量。 分而治之 分而治之(divide and conquer,D&C)是一種著名的遞歸式問題解決方法。只能解決一種問題的算法畢竟用處有限,而D&C提供了解決問題的思路,是另一個可供你使用的工具。 D&C...
摘要:在本書中你將學習比較不同算法的優(yōu)缺點該使用合并排序算法還是快速排序算法或者該使用數(shù)組還是鏈表。這樣的算法包括快速排序一種速度較快的排序算法。 在讀《算法圖解》這本書,這本書有兩個優(yōu)點: 手繪風格的圖,看著很讓人入戲; 算法采用Python語言描述,能更好的表達算法思想。 關于算法的學習有兩點心得: 算法思想最重要,理解了思想,算法是很容易寫出來的,所以盡量不要把過多精力放在細節(jié)上...
摘要:選擇排序是下一章將介紹的快速排序的基石。需要存儲多項數(shù)據(jù)時有兩種基本方式數(shù)組和鏈表。但它們并非都適用于所有的情形因此知道它們的差別很重要。選擇排序是一種靈巧的算法但其速度不是很快。 選擇排序是下一章將介紹的快速排序的基石。 內(nèi)存的工作原理 計算機就像是很多抽屜的集合體,每個抽屜都有地址。showImg(https://segmentfault.com/img/bVbpWvv?w=413...
摘要:解決最短路徑問題的算法被稱為廣度優(yōu)先搜索。廣度優(yōu)先搜索算法最早由年在如何從迷宮中尋找出路這一問題中提出。廣度優(yōu)先搜索讓你能夠找出兩樣東西之間的最短距離。使用廣度優(yōu)先搜索解決問題。 你經(jīng)常需要解決最短路徑問題(shorterst-path problem)。解決最短路徑問題的算法被稱為廣度優(yōu)先搜索。廣度優(yōu)先搜索算法最早由Edward F. Moore 1959年在如何從迷宮中尋找出路這一...
閱讀 2132·2021-11-19 09:58
閱讀 1719·2021-11-15 11:36
閱讀 2879·2019-08-30 15:54
閱讀 3399·2019-08-29 15:07
閱讀 2771·2019-08-26 11:47
閱讀 2825·2019-08-26 10:11
閱讀 2511·2019-08-23 18:22
閱讀 2759·2019-08-23 17:58