Python深入學(xué)習(xí)之閉包
閉包(closure)是函數(shù)式編程的重要的語(yǔ)法結(jié)構(gòu)。函數(shù)式編程是一種編程范式 (而面向過(guò)程編程和面向?qū)ο缶幊桃捕际蔷幊谭妒?。在面向過(guò)程編程中,我們見(jiàn)到過(guò)函數(shù)(function);在面向?qū)ο缶幊讨校覀円?jiàn)過(guò)對(duì)象(object)。函數(shù)和對(duì)象的根本目的是以某種邏輯方式組織代碼,并提高代碼的可重復(fù)使用性(reusability)。閉包也是一種組織代碼的結(jié)構(gòu),它同樣提高了代碼的可重復(fù)使用性。
不同的語(yǔ)言實(shí)現(xiàn)閉包的方式不同。Python以函數(shù)對(duì)象為基礎(chǔ),為閉包這一語(yǔ)法結(jié)構(gòu)提供支持的 (我們?cè)谔厥夥椒ㄅc多范式中,已經(jīng)多次看到Python使用對(duì)象來(lái)實(shí)現(xiàn)一些特殊的語(yǔ)法)。Python一切皆對(duì)象,函數(shù)這一語(yǔ)法結(jié)構(gòu)也是一個(gè)對(duì)象。在函數(shù)對(duì)象中,我們像使用一個(gè)普通對(duì)象一樣使用函數(shù)對(duì)象,比如更改函數(shù)對(duì)象的名字,或者將函數(shù)對(duì)象作為參數(shù)進(jìn)行傳遞。
函數(shù)對(duì)象的作用域
和其他對(duì)象一樣,函數(shù)對(duì)象也有其存活的范圍,也就是函數(shù)對(duì)象的作用域。函數(shù)對(duì)象是使用def語(yǔ)句定義的,函數(shù)對(duì)象的作用域與def所在的層級(jí)相同。比如下面代碼,我們?cè)趌ine_conf函數(shù)的隸屬范圍內(nèi)定義的函數(shù)line,就只能在line_conf的隸屬范圍內(nèi)調(diào)用。
def line_conf():
def line(x):
return 2*x+1
print(line(5)) # within the scope
line_conf()
print(line(5)) # out of the scope
line函數(shù)定義了一條直線(y = 2x + 1)??梢钥吹?,在line_conf()中可以調(diào)用line函數(shù),而在作用域之外調(diào)用line將會(huì)有下面的錯(cuò)誤:
NameError: name 'line' is not defined
說(shuō)明這時(shí)已經(jīng)在作用域之外。
同樣,如果使用lambda定義函數(shù),那么函數(shù)對(duì)象的作用域與lambda所在的層級(jí)相同。
閉包
函數(shù)是一個(gè)對(duì)象,所以可以作為某個(gè)函數(shù)的返回結(jié)果。
def line_conf():
def line(x):
return 2*x+1
return line # return a function object
my_line = line_conf()
print(my_line(5))
上面的代碼可以成功運(yùn)行。line_conf的返回結(jié)果被賦給line對(duì)象。上面的代碼將打印11。
如果line()的定義中引用了外部的變量,會(huì)發(fā)生什么呢?
def line_conf():
b = 15
def line(x):
return 2*x+b
return line # return a function object
b = 5
my_line = line_conf()
print(my_line(5))
我們可以看到,line定義的隸屬程序塊中引用了高層級(jí)的變量b,但b信息存在于line的定義之外 (b的定義并不在line的隸屬程序塊中)。我們稱b為line的環(huán)境變量。事實(shí)上,line作為line_conf的返回值時(shí),line中已經(jīng)包括b的取值(盡管b并不隸屬于line)。
上面的代碼將打印25,也就是說(shuō),line所參照的b值是函數(shù)對(duì)象定義時(shí)可供參考的b值,而不是使用時(shí)的b值。
一個(gè)函數(shù)和它的環(huán)境變量合在一起,就構(gòu)成了一個(gè)閉包(closure)。在Python中,所謂的閉包是一個(gè)包含有環(huán)境變量取值的函數(shù)對(duì)象。環(huán)境變量取值被保存在函數(shù)對(duì)象的__closure__屬性中。比如下面的代碼:
def line_conf():
b = 15
def line(x):
return 2*x+b
return line # return a function object
b = 5
my_line = line_conf()
print(my_line.__closure__)
print(my_line.__closure__[0].cell_contents)
__closure__里包含了一個(gè)元組(tuple)。這個(gè)元組中的每個(gè)元素是cell類型的對(duì)象。我們看到第一個(gè)cell包含的就是整數(shù)15,也就是我們創(chuàng)建閉包時(shí)的環(huán)境變量b的取值。
下面看一個(gè)閉包的實(shí)際例子:
def line_conf(a, b):
def line(x):
return ax + b
return line
line1 = line_conf(1, 1)
line2 = line_conf(4, 5)
print(line1(5), line2(5))
這個(gè)例子中,函數(shù)line與環(huán)境變量a,b構(gòu)成閉包。在創(chuàng)建閉包的時(shí)候,我們通過(guò)line_conf的參數(shù)a,b說(shuō)明了這兩個(gè)環(huán)境變量的取值,這樣,我們就確定了函數(shù)的最終形式(y = x + 1和y = 4x + 5)。我們只需要變換參數(shù)a,b,就可以獲得不同的直線表達(dá)函數(shù)。由此,我們可以看到,閉包也具有提高代碼可復(fù)用性的作用。
如果沒(méi)有閉包,我們需要每次創(chuàng)建直線函數(shù)的時(shí)候同時(shí)說(shuō)明a,b,x。這樣,我們就需要更多的參數(shù)傳遞,也減少了代碼的可移植性。利用閉包,我們實(shí)際上創(chuàng)建了泛函。line函數(shù)定義一種廣泛意義的函數(shù)。這個(gè)函數(shù)的一些方面已經(jīng)確定(必須是直線),但另一些方面(比如a和b參數(shù)待定)。隨后,我們根據(jù)line_conf傳遞來(lái)的參數(shù),通過(guò)閉包的形式,將最終函數(shù)確定下來(lái)。
閉包與并行運(yùn)算
閉包有效的減少了函數(shù)所需定義的參數(shù)數(shù)目。這對(duì)于并行運(yùn)算來(lái)說(shuō)有重要的意義。在并行運(yùn)算的環(huán)境下,我們可以讓每臺(tái)電腦負(fù)責(zé)一個(gè)函數(shù),然后將一臺(tái)電腦的輸出和下一臺(tái)電腦的輸入串聯(lián)起來(lái)。最終,我們像流水線一樣工作,從串聯(lián)的電腦集群一端輸入數(shù)據(jù),從另一端輸出數(shù)據(jù)。這樣的情境最適合只有一個(gè)參數(shù)輸入的函數(shù)。閉包就可以實(shí)現(xiàn)這一目的。
并行運(yùn)算正稱為一個(gè)熱點(diǎn)。這也是函數(shù)式編程又熱起來(lái)的一個(gè)重要原因。函數(shù)式編程早在1950年代就已經(jīng)存在,但應(yīng)用并不廣泛。然而,我們上面描述的流水線式的工作并行集群過(guò)程,正適合函數(shù)式編程。由于函數(shù)式編程這一天然優(yōu)勢(shì),越來(lái)越多的語(yǔ)言也開(kāi)始加入對(duì)函數(shù)式編程范式的支持。
相關(guān)文章
pycharm遠(yuǎn)程連接docker容器的操作流程
這篇文章主要給大家介紹了pycharm遠(yuǎn)程連接docker容器的操作流程,文中通過(guò)代碼示例和圖文講解介紹的非常詳細(xì),具有一定的參考價(jià)值,需要的朋友可以參考下2023-08-08Python實(shí)現(xiàn)優(yōu)先級(jí)隊(duì)列結(jié)構(gòu)的方法詳解
優(yōu)先級(jí)隊(duì)列(priority queue)是0個(gè)或多個(gè)元素的集合,每個(gè)元素都有一個(gè)優(yōu)先權(quán),接下來(lái)就來(lái)看一下簡(jiǎn)潔的Python實(shí)現(xiàn)優(yōu)先級(jí)隊(duì)列結(jié)構(gòu)的方法詳解:2016-06-06Python3.5集合及其常見(jiàn)運(yùn)算實(shí)例詳解
這篇文章主要介紹了Python3.5集合及其常見(jiàn)運(yùn)算,結(jié)合實(shí)例形式分析了Python3.5集合的定義、功能、交集、并集、差集等常見(jiàn)操作技巧與相關(guān)注意事項(xiàng),需要的朋友可以參考下2019-05-05Python實(shí)現(xiàn)扣除個(gè)人稅后的工資計(jì)算器示例
這篇文章主要介紹了Python實(shí)現(xiàn)扣除個(gè)人稅后的工資計(jì)算器,涉及Python流程控制與數(shù)學(xué)運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下2018-03-03Pytorch保存模型用于測(cè)試和用于繼續(xù)訓(xùn)練的區(qū)別詳解
今天小編就為大家分享一篇Pytorch保存模型用于測(cè)試和用于繼續(xù)訓(xùn)練的區(qū)別詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-01-01使用Python從零開(kāi)始擼一個(gè)區(qū)塊鏈
對(duì)數(shù)字貨幣的崛起感到新奇的我們,并且想知道其背后的技術(shù)——區(qū)塊鏈?zhǔn)窃鯓訉?shí)現(xiàn)的。這篇文章主要介紹了使用Python從零開(kāi)始擼一個(gè)區(qū)塊鏈,需要的朋友可以參考下2018-03-03Python:type、object、class與內(nèi)置類型實(shí)例
今天小編就為大家分享一篇Python:type、object、class與內(nèi)置類型實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-12-12基于Python第三方插件實(shí)現(xiàn)西游記章節(jié)標(biāo)注漢語(yǔ)拼音的方法
這篇文章主要介紹了基于Python第三方插件實(shí)現(xiàn)西游記章節(jié)標(biāo)注漢語(yǔ)拼音的方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-05-05