遞歸函數(shù)
閱讀目錄
楔子
在講今天的內(nèi)容之前,我們先來講一個故事,講的什么呢?從前有座山,山里有座廟,廟里有個老和尚講故事,講的什么呢?從前有座山,山里有座廟,廟里有個老和尚講故事,講的什么呢?從前有座山,山里有座廟,廟里有個老和尚講故事,講的什么呢?從前有座山,山里有座廟,廟里有個老和尚講故事,講的什么呢......這個故事你們不喊停我能講一天!我們說,生活中的例子也能被寫成程序,剛剛這個故事,讓你們寫,你們怎么寫呀?
while True:
? ? story = "? ? 從前有個山,山里有座廟,廟里老和尚講故事,
? ? 講的什么呢??
? ? "? ? print(story)
你肯定是要這么寫的,但是,現(xiàn)在我們已經(jīng)學了函數(shù)了,什么東西都要放到函數(shù)里去調(diào)用、執(zhí)行。于是你肯定會說,我就這么寫:
def story():
? ? s = """
? ? 從前有個山,山里有座廟,廟里老和尚講故事,
? ? 講的什么呢?
? ? """? ? print(s)
? ? while True:
? ? story()
但是大家來看看,我是怎么寫的!
def story():
? ? s = """
? ? 從前有個山,山里有座廟,廟里老和尚講故事,
? ? 講的什么呢?
? ? """? ? print(s)
? ? story()
story()
先不管函數(shù)最后的報錯,除了報錯之外,我們能看的出來,這一段代碼和上面的代碼執(zhí)行效果是一樣的。
初識遞歸
遞歸的定義——在一個函數(shù)里再調(diào)用這個函數(shù)本身
現(xiàn)在我們已經(jīng)大概知道剛剛講的story函數(shù)做了什么,就是在一個函數(shù)里再調(diào)用這個函數(shù)本身,這種魔性的使用函數(shù)的方式就叫做遞歸。
剛剛我們就已經(jīng)寫了一個最簡單的遞歸函數(shù)。
遞歸的最大深度——997
正如你們剛剛看到的,遞歸函數(shù)如果不受到外力的阻止會一直執(zhí)行下去。但是我們之前已經(jīng)說過關(guān)于函數(shù)調(diào)用的問題,每一次函數(shù)調(diào)用都會產(chǎn)生一個屬于它自己的名稱空間,如果一直調(diào)用下去,就會造成名稱空間占用太多內(nèi)存的問題,于是python為了杜絕此類現(xiàn)象,強制的將遞歸層數(shù)控制在了997(只要997!你買不了吃虧,買不了上當...).
拿什么來證明這個“997理論”呢?這里我們可以做一個實驗:
def foo(n):
? ? print(n)
? ? n += 1
? ? foo(n)
foo(1)
由此我們可以看出,未報錯之前能看到的最大數(shù)字就是997.當然了,997是python為了我們程序的內(nèi)存優(yōu)化所設定的一個默認值,我們當然還可以通過一些手段去修改它:
import sysprint(sys.setrecursionlimit(100000))
我們可以通過這種方式來修改遞歸的最大深度,剛剛我們將python允許的遞歸深度設置為了10w,至于實際可以達到的深度就取決于計算機的性能了。不過我們還是不推薦修改這個默認的遞歸深度,因為如果用997層遞歸都沒有解決的問題要么是不適合使用遞歸來解決要么是你代碼寫的太爛了~~~
看到這里,你可能會覺得遞歸也并不是多么好的東西,不如while True好用呢!然而,江湖上流傳這這樣一句話叫做:人理解循環(huán),神理解遞歸。所以你可別小看了遞歸函數(shù),很多人被攔在大神的門檻外這么多年,就是因為沒能領(lǐng)悟遞歸的真諦。而且之后我們學習的很多算法都會和遞歸有關(guān)系。來吧,只有學會了才有資本嫌棄!
再談遞歸
這里我們又要舉個例子來說明遞歸能做的事情。
例一:
現(xiàn)在你們問我,alex老師多大了?我說我不告訴你,但alex比 egon 大兩歲。
你想知道alex多大,你是不是還得去問egon?egon說,我也不告訴你,但我比武sir大兩歲。
你又問武sir,武sir也不告訴你,他說他比金鑫大兩歲。
那你問金鑫,金鑫告訴你,他40了。。。
這個時候你是不是就知道了?alex多大?
1金鑫 40
2武sir 42
3egon 44
4alex 46
你為什么能知道的?
首先,你是不是問alex的年齡,結(jié)果又找到egon、武sir、金鑫,你挨個兒問過去,一直到拿到一個確切的答案,然后順著這條線再找回來,才得到最終alex的年齡。這個過程已經(jīng)非常接近遞歸的思想。我們就來具體的我分析一下,這幾個人之間的規(guī)律。
age(4) = age(3) + 2
age(3) = age(2) + 2
age(2) = age(1) + 2
age(1) = 40
那這樣的情況下,我們的函數(shù)應該怎么寫呢?
def age(n):
? ? if n == 1:
? ? ? ? return 40
? ? else:
? ? ? ? return age(n-1)+2print(age(4))
遞歸函數(shù)與三級菜單
menu = {
? ? '北京': {
? ? ? ? '海淀': {
? ? ? ? ? ? '五道口': {
? ? ? ? ? ? ? ? 'soho': {},
? ? ? ? ? ? ? ? '網(wǎng)易': {},
? ? ? ? ? ? ? ? 'google': {}
? ? ? ? ? ? },
? ? ? ? ? ? '中關(guān)村': {
? ? ? ? ? ? ? ? '愛奇藝': {},
? ? ? ? ? ? ? ? '汽車之家': {},
? ? ? ? ? ? ? ? 'youku': {},
? ? ? ? ? ? },
? ? ? ? ? ? '上地': {
? ? ? ? ? ? ? ? '百度': {},
? ? ? ? ? ? },
? ? ? ? },
? ? ? ? '昌平': {
? ? ? ? ? ? '沙河': {
? ? ? ? ? ? ? ? '老男孩': {},
? ? ? ? ? ? ? ? '北航': {},
? ? ? ? ? ? },
? ? ? ? ? ? '天通苑': {},
? ? ? ? ? ? '回龍觀': {},
? ? ? ? },
? ? ? ? '朝陽': {},
? ? ? ? '東城': {},
? ? },
? ? '上海': {
? ? ? ? '閔行': {
? ? ? ? ? ? "人民廣場": {
? ? ? ? ? ? ? ? '炸雞店': {}
? ? ? ? ? ? }
? ? ? ? },
? ? ? ? '閘北': {
? ? ? ? ? ? '火車戰(zhàn)': {
? ? ? ? ? ? ? ? '攜程': {}
? ? ? ? ? ? }
? ? ? ? },
? ? ? ? '浦東': {},
? ? },
? ? '山東': {},
}
1 def threeLM(dic): 2? ? while True: 3? ? ? ? for k in dic:print(k) 4? ? ? ? key = input('input>>').strip() 5? ? ? ? if key == 'b' or key == 'q':return key 6? ? ? ? elif key in dic.keys() and dic[key]: 7? ? ? ? ? ? ret = threeLM(dic[key]) 8? ? ? ? ? ? if ret == 'q': return 'q'9 10 threeLM(menu)
還記得之前寫過的三級菜單作業(yè)么?現(xiàn)在咱們用遞歸來寫一下~
l = [menu]while l:
? ? for key in l[-1]:print(key)
? ? k = input('input>>').strip()? # 北京? ? if k in l[-1].keys() and l[-1][k]:l.append(l[-1][k])
? ? elif k == 'b':l.pop()
? ? elif k == 'q':break