13.0、python遞迴函式
遞迴函式
閱讀目錄
楔子
在講今天的內容之前,我們先來講一個故事,講的什麼呢?從前有座山,山裡有座廟,廟裡有個老和尚講故事,講的什麼呢?從前有座山,山裡有座廟,廟裡有個老和尚講故事,講的什麼呢?從前有座山,山裡有座廟,廟裡有個老和尚講故事,講的什麼呢?從前有座山,山裡有座廟,廟裡有個老和尚講故事,講的什麼呢……這個故事你們不喊停我能講一天!我們說,生活中的例子也能被寫成程式,剛剛這個故事,讓你們寫,你們怎麼寫呀?
while True:
story = ” 從前有個山,山裡有座廟,廟裡老和尚講故事,
講的什麼呢?
” print(story)
你肯定是要這麼寫的,但是,現在我們已經學了函式了,什麼東西都要放到函式裡去呼叫、執行。於是你肯定會說,我就這麼寫:
def story():
s = “””
從前有個山,山裡有座廟,廟裡老和尚講故事,
講的什麼呢?
“”” print(s)
while True:
story()
但是大家來看看,我是怎麼寫的!
def story():
s = “””
從前有個山,山裡有座廟,廟裡老和尚講故事,
講的什麼呢?
“”” print(s)
story()
story()
先不管函式最後的報錯,除了報錯之外,我們能看的出來,這一段程式碼和上面的程式碼執行效果是一樣的。
初識遞迴
遞迴的定義——在一個函式裡再呼叫這個函式本身
現在我們已經大概知道剛剛講的story函式做了什麼,就是在一個函式裡再呼叫這個函式本身,這種魔性的使用函式的方式就叫做遞迴。
剛剛我們就已經寫了一個最簡單的遞迴函式。
遞迴的最大深度——997
正如你們剛剛看到的,遞迴函式如果不受到外力的阻止會一直執行下去。但是我們之前已經說過關於函式呼叫的問題,每一次函式呼叫都會產生一個屬於它自己的名稱空間,如果一直呼叫下去,就會造成名稱空間佔用太多記憶體的問題,於是python為了杜絕此類現象,強制的將遞迴層數控制在了997(只要997!你買不了吃虧,買不了上當…).
拿什麼來證明這個“997理論”呢?這裡我們可以做一個實驗:
def foo(n):
print(n)
n += 1
foo(n)
foo(1)
由此我們可以看出,未報錯之前能看到的最大數字就是997.當然了,997是python為了我們程式的記憶體優化所設定的一個預設值,我們當然還可以通過一些手段去修改它:
import sysprint(sys.setrecursionlimit(100000))
我們可以通過這種方式來修改遞迴的最大深度,剛剛我們將python允許的遞迴深度設定為了10w,至於實際可以達到的深度就取決於計算機的效能了。不過我們還是不推薦修改這個預設的遞迴深度,因為如果用997層遞迴都沒有解決的問題要麼是不適合使用遞迴來解決要麼是你程式碼寫的太爛了~~~
看到這裡,你可能會覺得遞迴也並不是多麼好的東西,不如while True好用呢!然而,江湖上流傳這這樣一句話叫做:人理解迴圈,神理解遞迴。所以你可別小看了遞迴函式,很多人被攔在大神的門檻外這麼多年,就是因為沒能領悟遞迴的真諦。而且之後我們學習的很多演算法都會和遞迴有關係。來吧,只有學會了才有資本嫌棄!
再談遞迴
這裡我們又要舉個例子來說明遞迴能做的事情。
例一:
現在你們問我,alex老師多大了?我說我不告訴你,但alex比 egon 大兩歲。
你想知道alex多大,你是不是還得去問egon?egon說,我也不告訴你,但我比武sir大兩歲。
你又問武sir,武sir也不告訴你,他說他比金鑫大兩歲。
那你問金鑫,金鑫告訴你,他40了。。。
這個時候你是不是就知道了?alex多大?
1金鑫 40
2武sir 42
3egon 44
4alex 46
你為什麼能知道的?
首先,你是不是問alex的年齡,結果又找到egon、武sir、金鑫,你挨個兒問過去,一直到拿到一個確切的答案,然後順著這條線再找回來,才得到最終alex的年齡。這個過程已經非常接近遞迴的思想。我們就來具體的我分析一下,這幾個人之間的規律。
age(4) = age(3) + 2
age(3) = age(2) + 2
age(2) = age(1) + 2
age(1) = 40
那這樣的情況下,我們的函式應該怎麼寫呢?
def age(n):
if n == 1:
return 40
else:
return age(n-1)+2print(age(4))
遞迴函式與三級選單
menu = {
`北京`: {
`海淀`: {
`五道口`: {
`soho`: {},
`網易`: {},
`google`: {}
},
`中關村`: {
`愛奇藝`: {},
`汽車之家`: {},
`youku`: {},
},
`上地`: {
`百度`: {},
},
},
`昌平`: {
`沙河`: {
`老男孩`: {},
`北航`: {},
},
`天通苑`: {},
`回龍觀`: {},
},
`朝陽`: {},
`東城`: {},
},
`上海`: {
`閔行`: {
“人民廣場”: {
`炸雞店`: {}
}
},
`閘北`: {
`火車戰`: {
`攜程`: {}
}
},
`浦東`: {},
},
`山東`: {},
}
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)
還記得之前寫過的三級選單作業麼?現在我們們用遞迴來寫一下~
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
相關文章
- Python 函式進階-遞迴函式Python函式遞迴
- 1.5.6 python遞迴函式Python遞迴函式
- 遞迴函式遞迴函式
- 函式表示式–遞迴函式遞迴
- 函式的遞迴函式遞迴
- JavaScript 函式遞迴JavaScript函式遞迴
- php遞迴函式PHP遞迴函式
- 函式之遞迴函式遞迴
- 函式遞迴與生成式函式遞迴
- 遞迴函式的理解遞迴函式
- 從漢諾塔遊戲理解python遞迴函式遊戲Python遞迴函式
- 第 8 節:函式-匿名函式、遞迴函式函式遞迴
- JS函式表示式——函式遞迴、閉包JS函式遞迴
- day 17 – 1 遞迴函式遞迴函式
- 遞迴函式例項大全遞迴函式
- 遞迴函式-樹形列表遞迴函式
- 『無為則無心』Python函式 — 32、遞迴Python函式遞迴
- Python3之遞迴函式簡單示例Python遞迴函式
- 初學 PHP 函式的遞迴PHP函式遞迴
- C#語言函式遞迴C#函式遞迴
- GO語言————6.6 遞迴函式Go遞迴函式
- 直觀理解(尾)遞迴函式遞迴函式
- 好程式設計師Python教程系列遞迴函式與匿名函式呼叫程式設計師Python遞迴函式
- (011)我們一起學Python;匿名函式,遞迴函式Python函式遞迴
- Python遞迴函式,二分查詢演算法Python遞迴函式演算法
- 好程式設計師Python培訓分享Python的遞迴函式與匿名函式呼叫程式設計師Python遞迴函式
- 遞迴函式,可變引數列表遞迴函式
- JavaScript 中匿名函式的遞迴呼叫JavaScript函式遞迴
- 測開之函式進階· 第1篇《遞迴函式》函式遞迴
- SQL with as 的用法 以及遞迴函式的寫法 遞迴層次查詢SQL遞迴函式
- 課時22:函式:遞迴是神馬函式遞迴
- bilibiliclass10and11_函式遞迴函式遞迴
- 假如我們把函式都改成遞迴...函式遞迴
- sql 函式實現三種父子遞迴SQL函式遞迴
- 人人都能學會的python程式設計教程13:遞迴函式Python程式設計遞迴函式
- 遞迴、三元表示式、生成式(列表,字典)、匿名函式遞迴函式
- php 遞迴函式的三種實現方式PHP遞迴函式
- 編寫函式:遞迴求逆序 (Append Code) ★函式遞迴APP