網(wǎng)上寫遞歸的文章可以用汗牛充棟來形容了,大多數(shù)都非常清晰而又細(xì)致的角度上講解了遞歸的概念,原理等等。以前學(xué)生的時(shí)候,遞歸可以說一直是我的某種死穴,原理,細(xì)節(jié)我都懂,但是不管是在如何運(yùn)用或者如何試試算法題上真是有一種“聽過好多道理,依然過不好這一生的感覺”。經(jīng)常感覺信心受挫,力不從心吶。但是到后來如果不要去太糾結(jié)這些細(xì)節(jié),原理反而豁然開朗,突然我發(fā)現(xiàn)我可能是明白了。所以我的這篇瞎扯是想從一個(gè)宏觀的角度來扯扯遞歸算法,所以我起了這么個(gè)土洋結(jié)合的題目,因?yàn)槿驗(yàn)榈脑掞@得略裝b,但是我又實(shí)在找不到合適而又簡(jiǎn)潔的中文來表達(dá)“think in”的這個(gè)意思。 無論如何,希望看完這篇文章的人不再對(duì)遞歸感到混亂,也許能自己運(yùn)用遞歸解決算法問題或者實(shí)際問題,最重要的是希望能幫助一些曾經(jīng)和我有一樣困惑的人。
雖然是嘴上說的是想重點(diǎn)從宏觀上寫一些如何運(yùn)用遞歸,但是內(nèi)心還是想先扯一下遞歸的概念的。遞歸,我雖然沒查到他的最開始的出處,但是我感覺應(yīng)該不是從計(jì)算機(jī)這里創(chuàng)造出來的,這兩個(gè)字翻譯的也挺傳神,傳遞和歸約,但是如何用好這個(gè)傳遞和歸約就是過不好這一生的那一部分了。我一直覺得遞歸的思想頗有點(diǎn)“站在領(lǐng)導(dǎo)層”的感覺,為什么這么說,因?yàn)樵谠O(shè)計(jì)遞歸算法的時(shí)候,你只需要設(shè)計(jì)出大問題化小問題的遞歸算法,很多時(shí)候都是簡(jiǎn)單的幾個(gè)函數(shù)就能解決,剩下的具體都交給編譯器或者說語言本身來解決。但是正是這種特性,往往導(dǎo)致我們這種底層人民長(zhǎng)期的思維習(xí)慣在靈活運(yùn)用上會(huì)有點(diǎn)覺得“這尼瑪就行啦?”或者"還是有點(diǎn)不放心吶“這種感覺,正是這些感覺往往會(huì)導(dǎo)致一種混亂,從而舍本求末,造成在靈活運(yùn)用上的困難。所以,我一直覺得,在設(shè)計(jì)遞歸算法的時(shí)候,要有四步,第一先分析最簡(jiǎn)單的情況,第二,從小問題中總結(jié)大問題的規(guī)律,第三要寫出偽代碼,然后再寫真的代碼。
我會(huì)把遞歸問題分為三大類:
第一類,別細(xì)想,想多了絕逼能給你整懵圈型。 這類問題的有兩個(gè)特點(diǎn),一個(gè)是定睛一看,按普通算法想感覺完全一下子不知道從哪里下手,第二個(gè)就是當(dāng)你意識(shí)到這肯定得用遞歸啊,但是往往你會(huì)陷入一個(gè)怪圈,在你想出遞歸算法之后,你會(huì)自然的去驗(yàn)證。關(guān)鍵是,在你演這個(gè)時(shí)候或者試著仔細(xì)分析這個(gè)題目的時(shí)候會(huì)發(fā)現(xiàn)腦子越來越亂,越來越不夠使,最終本來想的透徹?zé)o比的問題就剩文字本身是清晰的了。這類問題比如想這種:
“一個(gè)有n階的樓梯,每次可以選擇上一階或者兩階,請(qǐng)問有多少種登頂?shù)姆椒???這個(gè)問題是一個(gè)爛大街的遞歸問題,如果你不用遞歸的思維去想,會(huì)覺得這玩意兒應(yīng)該怎么弄?但是這個(gè)問題使用遞歸的思維大問題化小問題其實(shí)很容易就想出解法。先想一階樓梯,兩階樓梯,三階樓梯試試,寫出偽代碼/步驟試試:
1. 如果只有一個(gè)階梯,只有一種方法,就是一次性上一階,直接登頂,應(yīng)該返回1
2. 如果有兩個(gè)階梯,兩種辦法,一次上一階,上兩次,或者一次直接上兩階,應(yīng)該返回2.
3. 如果有三個(gè)階梯,那么可以選擇1+1+1,1+2,2+1。
甚至你可以試試4,5,6階數(shù)的樓梯,但是你會(huì)發(fā)現(xiàn)的你的腦子到后面根本無法在繼續(xù)思考下去了,會(huì)有種大腦不夠用的感覺,這就到了該總結(jié)規(guī)律的時(shí)候,你會(huì)發(fā)現(xiàn)其實(shí)你上第n階樓梯的方法數(shù)目就等于你上第n-1階樓梯的方法數(shù)目加上上第n-2方法的數(shù)目,為什么?因?yàn)樵谶@兩種情況下,只需要一步你就可以登頂了,在方法的數(shù)目上你已經(jīng)沒得選了。到這里,你會(huì)忍不住的想去從細(xì)節(jié)上證明你的想法,別控制,你試試按照你的思維走下去。你會(huì)想,我靠,這么簡(jiǎn)單,真的嗎?我來想想到n-1階的時(shí)候是怎樣的呢?你會(huì)發(fā)現(xiàn)很快你就會(huì)到我上面的那個(gè)懵圈狀態(tài),反過來你會(huì)懷疑你的算法是不是對(duì)的,這樣你就會(huì)掛了。所以,試著僅僅從數(shù)學(xué)或者宏觀的角度證明一下這個(gè)算法,相信自己也相信計(jì)算機(jī)。所以這個(gè)問題的代碼很簡(jiǎn)單,就這么幾行:
climbStairs
就是這樣,很多時(shí)候用遞歸的方式解決問題寫出的代碼短的超乎想象,所以恐怕這也是造成自我懷疑的一個(gè)原因吧。為什么會(huì)造成懵圈,我覺得是我們大腦的堆棧不夠大,相比于計(jì)算機(jī),在不大的問題規(guī)模上已經(jīng)overflow了。
類似這樣的稍微復(fù)雜但是差不太多的還有:有無限個(gè)25美分,10美分,5美分和1美分的硬幣,如何組合成n美分,有多少種兌換方法? 可以按照我瞎扯的辦法試試,別細(xì)想,專注于宏觀設(shè)計(jì)出的算法本身,嘿嘿。
第二類,我覺得應(yīng)該叫”遞歸算法不容易想到“型。 這些問題設(shè)計(jì)出遞歸算法再反推回去不會(huì)造成腦子的懵圈, 但是想出這個(gè)遞歸算法容易導(dǎo)致自信心崩潰之類,因?yàn)檫@種問題一般遞歸解法都不太明顯,比如這個(gè)吧:
”反轉(zhuǎn)一個(gè)字符串,abcdefg,輸出gfedcba“,又是一個(gè)非常常見的問題,這個(gè)問題不是很難就能設(shè)計(jì)出一個(gè)一般的解法。利用一個(gè)循環(huán),計(jì)算好坐標(biāo),利用一個(gè)中間變量,相互交換字符的位置,大功告成。但是這個(gè)問題完全可以換一種思維,用遞歸的方法去解決。還是先從最小的規(guī)模先試試,還是得先寫下來:
1. 只有一個(gè)字符,直接輸出。
2. 有兩個(gè)字符,交換兩個(gè)字符的位置,輸出。
3. 有三個(gè)字符,中間一個(gè)字符不變,交換兩邊的字符,輸出。
4. 有四個(gè)字符,前兩個(gè)字符互相交換,后兩改字符互相交換,然后兩部分再兩兩交換,輸出。
5. 有五個(gè)字符,中間一個(gè)字符不不變,其余的重復(fù)4.
這個(gè)算法用寫出每一部分的方法很難總結(jié)出規(guī)律,但是如果真的寫出五個(gè)字符,在紙上試試,其實(shí)很容易就能找到分別交換兩個(gè)部分再互相交換的規(guī)律。這可能就是這里面最難的”算法設(shè)計(jì)“的這個(gè)部分吧。所以這個(gè)問題寫成代碼大概應(yīng)該是這樣的:
reverseString
這種問題,一般會(huì)沮喪在想出這個(gè)算法上,但是我覺得其實(shí)對(duì)于我們這種普通人來說,計(jì)算機(jī)里的算法設(shè)計(jì)多還是停留在多見世面才能解決問題的層面。畢竟那種能獨(dú)立設(shè)計(jì),思考出一個(gè)算法的人鳳毛麟角,所以,其實(shí)這個(gè)時(shí)候不必沮喪和失去信心,你現(xiàn)在不知道怎么做可能僅僅是見到的太少,你要見多了,大部分問題都能解決。
第三類,我把它叫”遞歸才是最好的理解答案思路型“,這種問題最常見于樹啊,圖啊之類的問題,簡(jiǎn)直不甚枚舉。特點(diǎn)是,這種問題你會(huì)發(fā)現(xiàn)你會(huì)自然的用遞歸的思想去思考這個(gè)問題,所以說,最后的代碼如果是以遞歸的形式呈現(xiàn)會(huì)跟符合人本身的自然思路。 隨便舉一個(gè)比較簡(jiǎn)單的例子:
”計(jì)算一個(gè)二叉樹所有左葉子節(jié)點(diǎn)的權(quán)重值的和"??粗@個(gè)問題思考,思路很容易就流淌出來,一個(gè)二叉樹所有左葉子節(jié)點(diǎn)權(quán)重值的和就等于一個(gè)左子樹的左葉子節(jié)點(diǎn)的權(quán)重值加上右子樹的左葉子節(jié)點(diǎn)的權(quán)重值?!斑f”的部分很容易就想出來了,那么“歸”的部分就可以從最小的問題思考一下,因?yàn)椤皻w”應(yīng)該滿足最小的問題集合,假設(shè)這個(gè)樹只有一個(gè)根節(jié)點(diǎn),那么可能返回0,如果是一個(gè)根節(jié)點(diǎn)帶一個(gè)左葉子節(jié)點(diǎn),那么應(yīng)該返回這個(gè)左葉子節(jié)點(diǎn)的值,因?yàn)槭亲笕~子節(jié)點(diǎn)的值的和,所以所有的右子樹在這里有可以化為另一個(gè)“遞”。貌似有點(diǎn)亂了,列一下可能能清晰一點(diǎn):
1. 如果只有根節(jié)點(diǎn),返回0。
2. 如果根節(jié)點(diǎn)有左葉子節(jié)點(diǎn),返回左葉子節(jié)點(diǎn)的值。
3. 繼續(xù)遍歷根節(jié)點(diǎn)的右子樹。
4. 遍歷所有當(dāng)前的左子樹和右子樹,重復(fù)1-3。
這樣再寫成代碼就很容易了:
sumOfLeftLeaves
這類問題你會(huì)發(fā)現(xiàn)遞歸寫出來的代碼更符合人的自然思維邏輯,比起那些傳統(tǒng)方法可能更容易展開和理解。
好了,上面就是我的一些胡扯,其實(shí)就像開頭說的,遞歸主要是"遞“和”歸“,先從宏觀的方面找到傳遞的路子,再用最小的問題集合找到歸約的條件和返回,大部分遞歸問題都很很容易能想出來。如果讓我選一個(gè)例子來最初步的理解遞歸算法,我會(huì)十分推薦快速排序,你可以就看一遍快速排序的算法描述,然后把它實(shí)現(xiàn)出來。不要小看實(shí)現(xiàn)一段某某算法,作為工程師,我覺得實(shí)現(xiàn)某種算法或者功能比設(shè)計(jì)算法更符合本職工作,也是一種非常大的能力。就像造汽車的和賽車手,造汽車的不一定開的好車,但是肯定會(huì)開車。賽車手大部分都不能獨(dú)自制造發(fā)動(dòng)機(jī),但是肯定懂點(diǎn)基本原理。所以說想不出來算法也并沒有什么好沮喪的。
最后,希望這篇文章能幫助需要的人。