在Python語言中,可以在函數(shù)中定義函數(shù)。 這種在函數(shù)中嵌套定義的函數(shù)也叫內(nèi)部函數(shù)。我們來看下面的代碼:
創(chuàng)新互聯(lián)公司專注于鄄城企業(yè)網(wǎng)站建設(shè),響應(yīng)式網(wǎng)站設(shè)計(jì),商城開發(fā)。鄄城網(wǎng)站建設(shè)公司,為鄄城等地區(qū)提供建站服務(wù)。全流程按需求定制網(wǎng)站,專業(yè)設(shè)計(jì),全程項(xiàng)目跟蹤,創(chuàng)新互聯(lián)公司專業(yè)和態(tài)度為您提供的服務(wù)
上述代碼中,定義了函數(shù)greet,在函數(shù)greet內(nèi)部又定義了一個(gè)函數(shù)inner_func, 并調(diào)用該函數(shù)打印了一串字符。
我們可以看到,內(nèi)部函數(shù)inner_func的定義和使用與普通函數(shù)基本相同。需要注意的是變量的作用域,在上述代碼中,函數(shù)參數(shù)name對(duì)于全局函數(shù)greet是局部變量,對(duì)內(nèi)部函數(shù)inner_func來說則是非局部變量。內(nèi)部函數(shù)對(duì)于非局部變量的訪問規(guī)則類似于標(biāo)準(zhǔn)的外部函數(shù)訪問全局變量。
從這個(gè)例子我們還可以看到內(nèi)部函數(shù)的一個(gè)作用,就是通過定義內(nèi)部函數(shù)的方式將一些功能隱藏起來,防止外部直接調(diào)用。常見的場(chǎng)景是,在一個(gè)復(fù)雜邏輯的函數(shù)中,將一些小的任務(wù)定義成內(nèi)部函數(shù),然后由這個(gè)外層函數(shù)使用,這樣可以使代碼更為清晰,易于維護(hù)。這些內(nèi)部函數(shù)只會(huì)在這個(gè)外層函數(shù)中使用,不能被其他函數(shù)或模塊使用。
在Python語言中, 函數(shù)也是對(duì)象,它可以被創(chuàng)建、賦值給變量,或者作為函數(shù)的返回值。我們來看下面這個(gè)例子。
在上述代碼中,在函數(shù)gen_greet內(nèi)部定義了inner_func函數(shù),并返回了一個(gè)inner_func函數(shù)對(duì)象。外部函數(shù)gen_greet返回了一個(gè)函數(shù)對(duì)象,所以像gen_greet這樣的函數(shù)也叫工廠函數(shù)。
在內(nèi)部函數(shù)inner_func中,使用了外部函數(shù)的傳參greet_words(非局部變量),以及函數(shù)的參數(shù)name(局部變量),來打印一個(gè)字符串。
接下來,調(diào)用gen_greet("Hello")創(chuàng)建一個(gè)函數(shù)對(duì)象say_hello,緊接著調(diào)用say_hello("Mr. Zhang"),輸出的結(jié)果為:Hello, Mr. Zhang!
同樣的,調(diào)用gen_greet("Hi")創(chuàng)建一個(gè)函數(shù)對(duì)象say_hi,調(diào)用say_hello("Mr. Zhang"),輸出的結(jié)果為:Hi,Tony!
我們可以發(fā)現(xiàn),gen_greet返回的函數(shù)對(duì)象具有記憶功能,它能夠把所需使用的非局部變量保存下來,用于后續(xù)被調(diào)用的時(shí)候使用。這種保存了非局部變量的函數(shù)對(duì)象被稱作閉包(closure)。
那么閉包是如何實(shí)現(xiàn)的呢?其實(shí)并不復(fù)雜,函數(shù)對(duì)象中有一個(gè)屬性__closure__,它就是在創(chuàng)建函數(shù)對(duì)象時(shí)用來保存這些非局部變量的。
__closure__屬性是一個(gè)元組或者None類型。在上述代碼中,我們可以通過下面方式查看:
函數(shù)的嵌套所實(shí)現(xiàn)的功能大都可以通過定義類的方式來實(shí)現(xiàn),而且類是更加面向?qū)ο蟮拇a編寫方式。
嵌套函數(shù)的一個(gè)主要用途是實(shí)現(xiàn)函數(shù)的裝飾器。我們看下面的代碼:
在上述代碼中,logger函數(shù)返回函數(shù)with_logging,with_logging則是打印了函數(shù)func的名稱及傳入的參數(shù),然后調(diào)用func, 并將參數(shù)傳遞給func。其中的@wraps(func)語句用于復(fù)制函數(shù)func的名稱、注釋文檔、參數(shù)列表等等,使得with_logging函數(shù)具有被裝飾的函數(shù)func相同的屬性。
代碼中接下來用@logger對(duì)函數(shù)power_func進(jìn)行修飾,它的作用等同于下面的代碼:
可見,裝飾器@符其實(shí)就是上述代碼的精簡(jiǎn)寫法。
通過了解了嵌套函數(shù)和閉包的工作原理,我們?cè)谑褂眠^程中就能夠更加得心應(yīng)手了。
今天遇到同樣的問題,就來答一波吧
1,如果是在類中,那么就很簡(jiǎn)單了,類中的一個(gè)函數(shù)調(diào)用另一個(gè)函數(shù),只要在那個(gè)被調(diào)用的函數(shù)前加self即可(圖如下,詳細(xì)可以參考筆者博客),
2,如果不是在類中,(這是筆者遇到的問題),有一個(gè)簡(jiǎn)單的方法,如下sin_f函數(shù)調(diào)用sin函數(shù)(注:a=sin()不能寫到sin_f()函數(shù)下,會(huì)說a沒聲明就調(diào)用):
3,如果是已經(jīng)存在的包,那么調(diào)用包更簡(jiǎn)單了,(同樣可以參考筆者上面給的那個(gè)博客第四部分)
4,最后,更多關(guān)于python問題可以參考筆者的python教程筆記
嵌套函數(shù)在執(zhí)行時(shí)(而不是在定義時(shí))從父范圍中查找變量。
編譯函數(shù)主體,然后驗(yàn)證“自由”變量(未在函數(shù)本身中通過賦值定義),然后將其作為閉包單元綁定到函數(shù),并且代碼使用索引引用每個(gè)單元格。pet_function因此具有一個(gè)自由變量(cage),然后將其通過一個(gè)閉合單元引用,索引為0的閉合本身指向局部變量cage在get_petters功能。
當(dāng)你實(shí)際調(diào)用該函數(shù)時(shí),該閉包將用于在你調(diào)用該函數(shù)時(shí)查看cage周圍作用域中的值。問題就在這里。在你調(diào)用函數(shù)時(shí),該函數(shù)已經(jīng)完成了對(duì)其結(jié)果的計(jì)算。將在在執(zhí)行過程中的一些點(diǎn)局部變量分配各的,和字符串,但在功能的結(jié)束,包含了最后一個(gè)值。因此,當(dāng)你調(diào)用每個(gè)動(dòng)態(tài)返回的函數(shù)時(shí),就會(huì)得到打印的值。get_petterscage'cow''dog''cat'cage'cat''cat'
解決方法是不依賴閉包。你可以改用部分函數(shù),創(chuàng)建新的函數(shù)作用域或?qū)⒆兞拷壎殛P(guān)鍵字parameter的默認(rèn)值。
部分函數(shù)示例,使用functools.partial():
from functools import partialdef pet_function(cage=None):
print "Mary pets the " + cage.animal + "."yield (animal, partial(gotimes, partial(pet_function, cage=cage)))
創(chuàng)建一個(gè)新的范圍示例:
def scoped_cage(cage=None):
def pet_function():
print "Mary pets the " + cage.animal + "."
return pet_functionyield (animal, partial(gotimes, scoped_cage(cage)))
將變量綁定為關(guān)鍵字參數(shù)的默認(rèn)值:
def pet_function(cage=cage):
print "Mary pets the " + cage.animal + "."yield (animal, partial(gotimes, pet_function))
無需scoped_cage在循環(huán)中定義函數(shù),編譯僅進(jìn)行一次,而不是在循環(huán)的每次迭代中進(jìn)行。
我怎么就變成大神了【笑哭】
def?A(a):
#這個(gè)下面有個(gè)TAB,就是為了讓下面的語句跟著你定義的這個(gè)A函數(shù)
print('i\'m?A')
#這下面的縮進(jìn)是在A函數(shù)里定義一個(gè)B函數(shù)
def?B(b):
#到這里的縮進(jìn)就是B函數(shù)的范圍了
print('i\'m?b')
print('a+b=',a+b)
#由于不跟著B函數(shù)的縮進(jìn),所以下面的這個(gè)B是A函數(shù)的范圍
B(3)
print('Done!')
A(5)
#樓主才剛學(xué)幾天呀
因?yàn)樽詈蟮哪蔷鋜eturn nested。
tester()()會(huì)自動(dòng)調(diào)用它的返回值,而此時(shí)的返回值為nested,即def nested()這個(gè)函數(shù),所以自然而然執(zhí)行到了里面的print語句。
你可以試試把最后那就return nested改成其他的如return nestedxxx,再tester()()時(shí)就會(huì)報(bào)錯(cuò)了。
另外,在python里對(duì)于方法ester和nested是沒有tester().nested()這種用法的,所以這樣輸入肯定報(bào)錯(cuò)的,如果ester和nested是類(class)的話才有這種寫法。
希望對(duì)你有所幫助~~