Python遞歸函數(shù) 二分查找算法實(shí)現(xiàn)解析
一、初始遞歸
遞歸函數(shù):在一個函數(shù)里在調(diào)用這個函數(shù)本身。
遞歸的最大深度:998
正如你們剛剛看到的,遞歸函數(shù)如果不受到外力的阻止會一直執(zhí)行下去。但是我們之前已經(jīng)說過關(guān)于函數(shù)調(diào)用的問題,每一次函數(shù)調(diào)用都會產(chǎn)生一個屬于它自己的名稱空間,如果一直調(diào)用下去,就會造成名稱空間占用太多內(nèi)存的問題,于是python為了杜絕此類現(xiàn)象,強(qiáng)制的將遞歸層數(shù)控制在了997(只要997!你買不了吃虧,買不了上當(dāng)...).
拿什么來證明這個“998理論”呢?這里我們可以做一個實(shí)驗(yàn):
def foo(n): print(n) n += 1 foo(n) foo(1)
由此我們可以看出,未報(bào)錯之前能看到的最大數(shù)字就是998.當(dāng)然了,997是python為了我們程序的內(nèi)存優(yōu)化所設(shè)定的一個默認(rèn)值,我們當(dāng)然還可以通過一些手段去修改它:
import sys print(sys.setrecursionlimit(100000))
我們可以通過這種方式來修改遞歸的最大深度,剛剛我們將python允許的遞歸深度設(shè)置為了10w,至于實(shí)際可以達(dá)到的深度就取決于計(jì)算機(jī)的性能了。不過我們還是不推薦修改這個默認(rèn)的遞歸深度,因?yàn)槿绻?97層遞歸都沒有解決的問題要么是不適合使用遞歸來解決要么是你代碼寫的太爛了~~~
看到這里,你可能會覺得遞歸也并不是多么好的東西,不如while True好用呢!然而,江湖上流傳這這樣一句話叫做:人理解循環(huán),神理解遞歸。所以你可別小看了遞歸函數(shù),很多人被攔在大神的門檻外這么多年,就是因?yàn)闆]能領(lǐng)悟遞歸的真諦。而且之后我們學(xué)習(xí)的很多算法都會和遞歸有關(guān)系。來吧,只有學(xué)會了才有資本嫌棄!
二、遞歸示例講解
這里我們又要舉個例子來說明遞歸能做的事情。
例一:
現(xiàn)在你們問我,alex老師多大了?我說我不告訴你,但alex比 egon 大兩歲。
你想知道alex多大,你是不是還得去問egon?egon說,我也不告訴你,但我比武sir大兩歲。
你又問武sir,武sir也不告訴你,他說他比太白大兩歲。
那你問太白,太白告訴你,他18了。
這個時候你是不是就知道了?alex多大?
| 1 | 金鑫 | 18 |
|---|---|---|
| 2 | 武sir | 20 |
| 3 | egon | 22 |
| 4 | alex | 24 |
你為什么能知道的?
首先,你是不是問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)+2
print(age(4))
如果有這樣一個列表,讓你從這個列表中找到66的位置,你要怎么做?
l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]
你說,so easy!
l.index(66)...
我們之所以用index方法可以找到,是因?yàn)閜ython幫我們實(shí)現(xiàn)了查找方法。如果,index方法不給你用了。。。你還能找到這個66么?
l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]
i = 0
for num in l:
if num == 66:
print(i)
i+=1
上面這個方法就實(shí)現(xiàn)了從一個列表中找到66所在的位置了。
但我們現(xiàn)在是怎么找到這個數(shù)的呀?是不是循環(huán)這個列表,一個一個的找的呀?假如我們這個列表特別長,里面好好幾十萬個數(shù),那我們找一個數(shù)如果運(yùn)氣不好的話是不是要對比十幾萬次?這樣效率太低了,我們得想一個新辦法。
二分查找算法
l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]
你觀察這個列表,這是不是一個從小到大排序的有序列表呀?
如果這樣,假如我要找的數(shù)比列表中間的數(shù)還大,是不是我直接在列表的后半邊找就行了?

這就是二分查找算法!
那么落實(shí)到代碼上我們應(yīng)該怎么實(shí)現(xiàn)呢?
簡單版二分法
l = [2,3,5,10,15,16,18,22,26,30,32,35,41,42,43,55,56,66,67,69,72,76,82,83,88]
def func(l,aim):
mid = (len(l)-1)//2
if l:
if aim > l[mid]:
func(l[mid+1:],aim)
elif aim < l[mid]:
func(l[:mid],aim)
elif aim == l[mid]:
print("bingo",mid)
else:
print('找不到')
func(l,66)
func(l,6)
升級版二分法
l1 = [1, 2, 4, 5, 7, 9]
def two_search(l,aim,start=0,end=None):
end = len(l)-1 if end is None else end
mid_index = (end - start) // 2 + start
if end >= start:
if aim > l[mid_index]:
return two_search(l,aim,start=mid_index+1,end=end)
elif aim < l[mid_index]:
return two_search(l,aim,start=start,end=mid_index-1)
elif aim == l[mid_index]:
return mid_index
else:
return '沒有此值'
else:
return '沒有此值'
print(two_search(l1,9))
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Python非單向遞歸函數(shù)如何返回全部結(jié)果
- Python通過遞歸函數(shù)輸出嵌套列表元素
- python遞歸函數(shù)求n的階乘,優(yōu)缺點(diǎn)及遞歸次數(shù)設(shè)置方式
- python函數(shù)局部變量、全局變量、遞歸知識點(diǎn)總結(jié)
- python中的函數(shù)遞歸和迭代原理解析
- 提升Python效率之使用循環(huán)機(jī)制代替遞歸函數(shù)
- 讓你Python到很爽的加速遞歸函數(shù)的裝飾器
- python遞歸函數(shù)繪制分形樹的方法
- Python進(jìn)階之遞歸函數(shù)的用法及其示例
- python基礎(chǔ)學(xué)習(xí)之遞歸函數(shù)知識總結(jié)
相關(guān)文章
python實(shí)現(xiàn)TCP文件接收發(fā)送
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)TCP文件接收發(fā)送,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09
Python基礎(chǔ)之?dāng)?shù)據(jù)類型知識匯總
今天帶大家復(fù)習(xí)一下Python基礎(chǔ)知識,文中對數(shù)據(jù)類型相關(guān)知識做了詳細(xì)的匯總,對剛?cè)腴Tpython的小伙伴很有幫助喲,需要的朋友可以參考下2021-05-05
jupyter運(yùn)行時左邊一直出現(xiàn)*號問題及解決
這篇文章主要介紹了jupyter運(yùn)行時左邊一直出現(xiàn)*號問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-09-09
python將xml xsl文件生成html文件存儲示例講解
這篇文章主要介紹了python將xml、xsl文件轉(zhuǎn)成html文件存儲方法,大家參考使用吧2013-12-12
Python異步爬蟲實(shí)現(xiàn)原理與知識總結(jié)
之前有很多小伙伴想看Python異步爬蟲的有關(guān)知識總結(jié),這次它來了,文中有非常詳細(xì)的代碼示例與注釋,即使對剛開始學(xué)python的小伙伴也很友好,,需要的朋友可以參考下2021-05-05
使用NumPy進(jìn)行數(shù)組數(shù)據(jù)處理的示例詳解
NumPy是Python中用于數(shù)值計(jì)算的核心包之一,它提供了大量的高效數(shù)組操作函數(shù)和數(shù)學(xué)函數(shù),可以支持多維數(shù)組和矩陣運(yùn)算。本文主要為大家介紹了NumPy進(jìn)行數(shù)組數(shù)據(jù)處理的具體方法,需要的可以參考一下2023-03-03
如何使用Selenium實(shí)現(xiàn)簡單的網(wǎng)絡(luò)自動化操作指南
Selenium是一個用于Web應(yīng)用測試的工具,Selenium測試直接運(yùn)行在瀏覽器中,就像真正的用戶在操作一樣,這篇文章主要給大家介紹了關(guān)于如何使用Selenium實(shí)現(xiàn)簡單的網(wǎng)絡(luò)自動化操作的相關(guān)資料,需要的朋友可以參考下2024-03-03

