十年網(wǎng)站開發(fā)經(jīng)驗(yàn) + 多家企業(yè)客戶 + 靠譜的建站團(tuán)隊(duì)
量身定制 + 運(yùn)營(yíng)維護(hù)+專業(yè)推廣+無(wú)憂售后,網(wǎng)站問題一站解決
閉包?
站在用戶的角度思考問題,與客戶深入溝通,找到河津網(wǎng)站設(shè)計(jì)與河津網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗(yàn),讓設(shè)計(jì)與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個(gè)性化、用戶體驗(yàn)好的作品,建站類型包括:成都網(wǎng)站設(shè)計(jì)、做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣、域名與空間、網(wǎng)絡(luò)空間、企業(yè)郵箱。業(yè)務(wù)覆蓋河津地區(qū)。
1.函數(shù)引用
運(yùn)行結(jié)果:
圖解:
相關(guān)推薦:《Python視頻教程》
2.什么是閉包
運(yùn)行結(jié)果:
3.看一個(gè)閉包的實(shí)際例子:
運(yùn)行結(jié)果:
這個(gè)例子中,函數(shù)line與變量a,b構(gòu)成閉包。在創(chuàng)建閉包的時(shí)候,我們通過line_conf的參數(shù)a,b說明了這兩個(gè)變量的取值,這樣,我們就確定了函數(shù)的最終形式(y = x + 1和y = 4x + 5)。我們只需要變換參數(shù)a,b,就可以獲得不同的直線表達(dá)函數(shù)。由此,我們可以看到,閉包也具有提高代碼可復(fù)用性的作用。
如果沒有閉包,我們需要每次創(chuàng)建直線函數(shù)的時(shí)候同時(shí)說明a,b,x。這樣,我們就需要更多的參數(shù)傳遞,也減少了代碼的可移植性。
相關(guān)推薦:
Python中的迭代器是什么
一.閉包的定義:
在一個(gè)函數(shù)的內(nèi)部,再定義一個(gè)函數(shù)(內(nèi)部函數(shù))。這個(gè)內(nèi)部函數(shù)引用了外部函數(shù)的變量,并且外部函數(shù)返回這個(gè)內(nèi)部函數(shù), 我們把這個(gè)使用外部函數(shù)變量的內(nèi)部函數(shù)稱為 閉包 。
簡(jiǎn)而言之, 閉包就是能夠讀取外部函數(shù)內(nèi)的變量的函數(shù)。
例如:
形成閉包的兩個(gè)條件:
二.閉包的用途
① 可以讀取函數(shù)內(nèi)部的變量
② 將一些變量的值始終保存到內(nèi)存中
1.讀取函數(shù)內(nèi)部的變量
在一般情況下,在函數(shù)外部我們是不能訪問到函數(shù)內(nèi)部的變量的。但是, 有時(shí)想要在函數(shù)外部能夠訪問到函數(shù)內(nèi)部的變量,那么就可以使用閉包。
例如:
上面的代碼可以看出,print(a)會(huì)拋異常NameError: name 'a' is not defined。在函數(shù)f1的外面無(wú)法訪問它的變量的。
在函數(shù)f1里面定義一個(gè)閉包函數(shù)就可以訪問到了
例如:
2.將一些變量的值始終保存到內(nèi)存中
運(yùn)行結(jié)果:
通過上面的輸出結(jié)果可以看出閉包保存了外部函數(shù)內(nèi)的變量n1的值1,每次執(zhí)行閉包都是在n1 = 1 基礎(chǔ)上進(jìn)行計(jì)算的。
三.閉包的缺點(diǎn)
1. 由于閉包會(huì)使得函數(shù)中的變量都被保存在內(nèi)存中,會(huì)增加 內(nèi)存消耗 ,所以不能濫用閉包,否則會(huì)造成程序的性能問題,可能導(dǎo)致內(nèi)存泄露
2. 閉包無(wú)法改變外部函數(shù)局部變量指向的內(nèi)存地址
3. 返回閉包時(shí),返回函數(shù)不要引用任何循環(huán)變量,或者后續(xù)會(huì)發(fā)生變化的變量
四.判斷一個(gè)函數(shù)是否是閉包
判斷一個(gè)函數(shù)是不是閉包,可以查看它的 closure 屬性。如果該函數(shù)是閉包,查看該屬性將會(huì)返回一個(gè)cell對(duì)象組成的tuple。如果我們分別對(duì)每個(gè)cell對(duì)象查看其cell_contents屬性,返回的內(nèi)容就是閉包引用的自由變量的值。
運(yùn)行結(jié)果:
閉包的__closure__方法,可以展示出閉包儲(chǔ)存了外部函數(shù)的兩個(gè)變量,cell的內(nèi)存地址是什么,在cell里面儲(chǔ)存的對(duì)象類型是int,這個(gè)int儲(chǔ)存的內(nèi)存地址是什么。
閉包的__closure__方法,可以查看每個(gè)cell對(duì)象的內(nèi)容
運(yùn)行結(jié)果:
cell_contents解釋了局部變量在脫離函數(shù)后仍然可以在函數(shù)之外被訪問的原因,因?yàn)樽兞勘淮鎯?chǔ)在cell_contents中了。
在Python語(yǔ)言中,可以在函數(shù)中定義函數(shù)。 這種在函數(shù)中嵌套定義的函數(shù)也叫內(nèi)部函數(shù)。我們來(lái)看下面的代碼:
上述代碼中,定義了函數(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來(lái)說則是非局部變量。內(nèi)部函數(shù)對(duì)于非局部變量的訪問規(guī)則類似于標(biāo)準(zhǔn)的外部函數(shù)訪問全局變量。
從這個(gè)例子我們還可以看到內(nèi)部函數(shù)的一個(gè)作用,就是通過定義內(nèi)部函數(shù)的方式將一些功能隱藏起來(lái),防止外部直接調(diào)用。常見的場(chǎng)景是,在一個(gè)復(fù)雜邏輯的函數(shù)中,將一些小的任務(wù)定義成內(nèi)部函數(shù),然后由這個(gè)外層函數(shù)使用,這樣可以使代碼更為清晰,易于維護(hù)。這些內(nèi)部函數(shù)只會(huì)在這個(gè)外層函數(shù)中使用,不能被其他函數(shù)或模塊使用。
在Python語(yǔ)言中, 函數(shù)也是對(duì)象,它可以被創(chuàng)建、賦值給變量,或者作為函數(shù)的返回值。我們來(lái)看下面這個(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(局部變量),來(lái)打印一個(gè)字符串。
接下來(lái),調(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ì)象具有記憶功能,它能夠把所需使用的非局部變量保存下來(lái),用于后續(xù)被調(diào)用的時(shí)候使用。這種保存了非局部變量的函數(shù)對(duì)象被稱作閉包(closure)。
那么閉包是如何實(shí)現(xiàn)的呢?其實(shí)并不復(fù)雜,函數(shù)對(duì)象中有一個(gè)屬性__closure__,它就是在創(chuàng)建函數(shù)對(duì)象時(shí)用來(lái)保存這些非局部變量的。
__closure__屬性是一個(gè)元組或者None類型。在上述代碼中,我們可以通過下面方式查看:
函數(shù)的嵌套所實(shí)現(xiàn)的功能大都可以通過定義類的方式來(lái)實(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)語(yǔ)句用于復(fù)制函數(shù)func的名稱、注釋文檔、參數(shù)列表等等,使得with_logging函數(shù)具有被裝飾的函數(shù)func相同的屬性。
代碼中接下來(lái)用@logger對(duì)函數(shù)power_func進(jìn)行修飾,它的作用等同于下面的代碼:
可見,裝飾器@符其實(shí)就是上述代碼的精簡(jiǎn)寫法。
通過了解了嵌套函數(shù)和閉包的工作原理,我們?cè)谑褂眠^程中就能夠更加得心應(yīng)手了。
在函數(shù)中可以定義另一個(gè)函數(shù)時(shí),如果內(nèi)部的函數(shù)引用了外部的函數(shù)的變量,則可能產(chǎn)生閉包。
閉包可以用來(lái)在一個(gè)函數(shù)與一組私有變量之間創(chuàng)建關(guān)聯(lián)關(guān)系。
在給定函數(shù)被多次調(diào)用的過程中,這些私有變量能夠保持其持久性。
形成閉包的三個(gè)條件
必須有一個(gè)內(nèi)嵌函數(shù)—這對(duì)應(yīng)函數(shù)之間的嵌套;
內(nèi)嵌函數(shù)必須引用一個(gè)定義在閉合范圍內(nèi)的變量—內(nèi)部函數(shù)引用外部變量;
外部函數(shù)必須返回內(nèi)嵌函數(shù)—必須返回內(nèi)部函數(shù)。
換句話來(lái)說:閉包的概念很簡(jiǎn)單,一個(gè)可以引用在函數(shù)閉合范圍內(nèi)變量的函數(shù),即內(nèi)部函數(shù),只有那個(gè)內(nèi)部函數(shù)才有所謂的__closure__屬性。
閉包的原理
形成閉包之后,閉包函數(shù)會(huì)獲得一個(gè)非空的_Closure_屬性,這個(gè)屬性是一個(gè)元組。
組里面的對(duì)象為cell對(duì)象,而訪問cell對(duì)象的cell_contents屬性則可以得到閉包變量的當(dāng)前值。
而隨著閉包的繼續(xù)調(diào)用,變量會(huì)進(jìn)行再次更新。由此可見,一般形成閉包之后,Python確定會(huì)將_closure_和閉包函數(shù)綁定作為儲(chǔ)存閉包變量的場(chǎng)所。
閉包的好處是什么?
其實(shí),閉包并不是必須的。
沒有閉包的話,Python的功能不會(huì)受到任何影響;但有了閉包之后,可以提供一種額外的解決方案。
在python中,函數(shù)可以被嵌套定義,也就是說,函數(shù)中可以定義函數(shù)。該函數(shù)還可以將其內(nèi)部定義的函數(shù)作為返回值返回。
閉包的定義:一般來(lái)說,我們可以認(rèn)為,如果一個(gè)函數(shù)可以讀取其他函數(shù)中的局部變量,那么它們就構(gòu)成了閉包。
注意 :閉包的定義不是特別清晰,但大體上的意思是這樣的。
我們知道,普通的函數(shù)是可以使用全局變量的
類似的,函數(shù)中定義的函數(shù),也是可以使用外部函數(shù)的變量的。因此,滿足了函數(shù)讀取了其他函數(shù)局部變量的這一條件,他們因此構(gòu)成了閉包。
在閉包的使用中,我們可以先給外部的函數(shù)賦予不同的局部變量,然后再調(diào)用其中內(nèi)部的函數(shù)時(shí),就可以讀取到這些不同的局部變量了。
外部變量的使用 在普通函數(shù)中,雖然可以直接使用全局變量,但是不可以直接修改全局變量。從變量的作用域來(lái)說,一旦你嘗試修改全局變量,那么就會(huì)嘗試創(chuàng)建并使用一個(gè)同名的局部變量。因此,如果你需要在普通函數(shù)中修改全局變量,需要使用global
同樣的,如果你希望通過定義在內(nèi)部的函數(shù)去修改其外部函數(shù)的變量,那么必須使用nonlocal
閉包:python函數(shù)的內(nèi)部的變量離開這個(gè)函數(shù)就失去了作用域而不復(fù)存在
但是嵌套函數(shù)可以!
解析: x是屬于fn1函數(shù)的,但是在 fn1()調(diào)用結(jié)束后 我們又加了兩個(gè)括號(hào)調(diào)用到了fn3,你看他還是能輸出x的值
但嵌套函數(shù)只是引用它!不能修改它,要在嵌套函數(shù)里修改x的值需要申明nonlocal x
工廠函數(shù):
函數(shù)return的時(shí)候返回一個(gè)函數(shù)名
參考: 石溪的答案