摘要:閉包可以用來(lái)在一個(gè)函數(shù)與一組私有變量之間創(chuàng)建關(guān)聯(lián)關(guān)系。夾帶私貨外部變量返回的是函數(shù),帶私貨的函數(shù)支持將函數(shù)當(dāng)成對(duì)象使用的編程語(yǔ)言,一般都支持閉包。所以說(shuō)當(dāng)你的裝飾器需要自定義參數(shù)時(shí),一般都會(huì)形成閉包。
Python中的閉包不是一個(gè)一說(shuō)就能明白的概念,但是隨著你往學(xué)習(xí)的深入,無(wú)論如何你都需要去了解這么一個(gè)東西。
閉包的概念我們嘗試從概念上去理解一下閉包。
在一些語(yǔ)言中,在函數(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)用的過(guò)程中,這些私有變量能夠保持其持久性。
—— 維基百科)
用比較容易懂的人話說(shuō),就是當(dāng)某個(gè)函數(shù)被當(dāng)成對(duì)象返回時(shí),夾帶了外部變量,就形成了一個(gè)閉包。看例子。
def make_printer(msg): def printer(): print msg # 夾帶私貨(外部變量) return printer # 返回的是函數(shù),帶私貨的函數(shù) printer = make_printer("Foo!") printer()
支持將函數(shù)當(dāng)成對(duì)象使用的編程語(yǔ)言,一般都支持閉包。比如Python, JavaScript。
如何理解閉包閉包存在有什么意義呢?為什么需要閉包?
我個(gè)人認(rèn)為,閉包存在的意義就是它夾帶了外部變量(私貨),如果它不夾帶私貨,它和普通的函數(shù)就沒(méi)有任何區(qū)別。同一個(gè)的函數(shù)夾帶了不同的私貨,就實(shí)現(xiàn)了不同的功能。其實(shí)你也可以這么理解,閉包和面向接口編程的概念很像,可以把閉包理解成輕量級(jí)的接口封裝。
接口定義了一套對(duì)方法簽名的約束規(guī)則。
def tag(tag_name): def add_tag(content): return "<{0}>{1}{0}>".format(tag_name, content) return add_tag content = "Hello" add_tag = tag("a") print add_tag(content) # Hello add_tag = tag("b") print add_tag(content) # Hello
在這個(gè)例子里,我們想要一個(gè)給content加tag的功能,但是具體的tag_name是什么樣子的要根據(jù)實(shí)際需求來(lái)定,對(duì)外部調(diào)用的接口已經(jīng)確定,就是add_tag(content)。如果按照面向接口方式實(shí)現(xiàn),我們會(huì)先把add_tag寫(xiě)成接口,指定其參數(shù)和返回類型,然后分別去實(shí)現(xiàn)a和b的add_tag。
但是在閉包的概念中,add_tag就是一個(gè)函數(shù),它需要tag_name和content兩個(gè)參數(shù),只不過(guò)tag_name這個(gè)參數(shù)是打包帶走的。所以一開(kāi)始時(shí)就可以告訴我怎么打包,然后帶走就行。
上面的例子不太生動(dòng),其實(shí)在我們生活和工作中,閉包的概念也很常見(jiàn)。比如說(shuō)手機(jī)撥號(hào),你只關(guān)心電話打給誰(shuí),而不會(huì)去糾結(jié)每個(gè)品牌的手機(jī)是怎么實(shí)現(xiàn)的,用到了哪些模塊。再比如去餐館吃飯,你只要付錢就可以享受到服務(wù),你并不知道那桌飯菜用了多少地溝油。這些都可以看成閉包,返回來(lái)的是一些功能或者服務(wù)(打電話,用餐),但是這些功能使用了外部變量(天線,地溝油等等)。
你也可以把一個(gè)類實(shí)例看成閉包,當(dāng)你在構(gòu)造這個(gè)類時(shí),使用了不同的參數(shù),這些參數(shù)就是閉包里的包,這個(gè)類對(duì)外提供的方法就是閉包的功能。但是類遠(yuǎn)遠(yuǎn)大于閉包,因?yàn)殚]包只是一個(gè)可以執(zhí)行的函數(shù),但是類實(shí)例則有可能提供很多方法。
何時(shí)使用閉包其實(shí)閉包在Python中很常見(jiàn),只不過(guò)你沒(méi)特別注意這就是一個(gè)閉包。比如Python中的裝飾器Decorator,假如你需要寫(xiě)一個(gè)帶參數(shù)的裝飾器,那么一般都會(huì)生成閉包。
為什么?因?yàn)镻ython的裝飾器是一個(gè)固定的函數(shù)接口形式。它要求你的裝飾器函數(shù)(或裝飾器類)必須接受一個(gè)函數(shù)并返回一個(gè)函數(shù):
# how to define def wrapper(func1): # 接受一個(gè)callable對(duì)象 return func2 # 返回一個(gè)對(duì)象,一般為函數(shù) # how to use def target_func(args): # 目標(biāo)函數(shù) pass # 調(diào)用方式一,直接包裹 result = wrapper(target_func)(args) # 調(diào)用方式二,使用@語(yǔ)法,等同于方式一 @wrapper def target_func(args): pass result = target_func()
那么如果你的裝飾器如果帶參數(shù)呢?那么你就需要在原來(lái)的裝飾器上再包一層,用于接收這些參數(shù)。這些參數(shù)(私貨)傳遞到內(nèi)層的裝飾器里后,閉包就形成了。所以說(shuō)當(dāng)你的裝飾器需要自定義參數(shù)時(shí),一般都會(huì)形成閉包。(類裝飾器例外)
def html_tags(tag_name): def wrapper_(func): def wrapper(*args, **kwargs): content = func(*args, **kwargs) return "<{tag}>{content}{tag}>".format(tag=tag_name, content=content) return wrapper return wrapper_ @html_tags("b") def hello(name="Toby"): return "Hello {}!".format(name) # 不用@的寫(xiě)法如下 # hello = html_tag("b")(hello) # html_tag("b") 是一個(gè)閉包,它接受一個(gè)函數(shù),并返回一個(gè)函數(shù) print hello() # Hello Toby! print hello("world") # Hello world!
關(guān)于裝飾器的更深入剖析,可以看我寫(xiě)的另外一篇博客。
再深入一點(diǎn)其實(shí)也不必太深入,理解這上面的概念,很多看起來(lái)頭疼的代碼也不過(guò)如此。
下面讓我們來(lái)了解一下閉包的包到底長(zhǎng)什么樣子。其實(shí)閉包函數(shù)相對(duì)與普通函數(shù)會(huì)多出一個(gè)__closure__的屬性,里面定義了一個(gè)元組用于存放所有的cell對(duì)象,每個(gè)cell對(duì)象一一保存了這個(gè)閉包中所有的外部變量。
>>> def make_printer(msg1, msg2): def printer(): print msg1, msg2 return printer >>> printer = make_printer("Foo", "Bar") # 形成閉包 >>> printer.__closure__ # 返回cell元組 (, | ) >>> printer.__closure__[0].cell_contents # 第一個(gè)外部變量 "Foo" >>> printer.__closure__[1].cell_contents # 第二個(gè)外部變量 "Bar" |
原理就是這么簡(jiǎn)單。
參考鏈接https://www.the5fire.com/clos...
http://stackoverflow.com/ques...
關(guān)于作者:Python技術(shù)愛(ài)好者,目前從事測(cè)試開(kāi)發(fā)相關(guān)工作,轉(zhuǎn)載請(qǐng)注明原文出處。
歡迎關(guān)注我的博客 http://betacat.online,你可以到我的公眾號(hào)中去當(dāng)吃瓜群眾。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/44229.html
摘要:項(xiàng)目地址閉包在計(jì)算機(jī)科學(xué)中,閉包英語(yǔ),又稱詞法閉包或函數(shù)閉包,是引用了自由變量的函數(shù)。這個(gè)被引用的自由變量將和這個(gè)函數(shù)一同存在,即使已經(jīng)離開(kāi)了創(chuàng)造它的環(huán)境也不例外。 項(xiàng)目地址:https://git.io/pytips 閉包(Closure) 在計(jì)算機(jī)科學(xué)中,閉包(英語(yǔ):Closure),又稱詞法閉包(Lexical Closure)或函數(shù)閉包(function closures),是...
摘要:我們說(shuō)觸發(fā)了閉包的函數(shù)叫做閉包函數(shù)閉包最大的特點(diǎn)就是它可以被外層函數(shù)返回后賦值給一個(gè)變量,并且攜帶了外層函數(shù)內(nèi)定義的變量例子如下變量為函數(shù)開(kāi)辟的局部命名空間內(nèi)定義的變量函數(shù)內(nèi)引用了變量的內(nèi)層函數(shù)名被當(dāng)作返回值,此時(shí)閉包規(guī)則達(dá)成。 什么是閉包? 其實(shí)我們?cè)谑褂煤瘮?shù)過(guò)程中不經(jīng)意間就會(huì)觸發(fā)閉包,因?yàn)榭倳?huì)出于某種原因會(huì)在函數(shù)內(nèi)引用或修改上一層函數(shù)的變量,這時(shí)就會(huì)觸發(fā)閉包 那么什么是閉包?其實(shí)就...
摘要:在計(jì)算機(jī)科學(xué)中,閉包又稱詞法閉包或函數(shù)閉包,是引用了自由變量的函數(shù)。閉包被廣泛應(yīng)用于函數(shù)式語(yǔ)言中。運(yùn)用閉包可以避免對(duì)全局變量的使用。將棧頂?shù)脑厝〕觯瑒?chuàng)建元組,并將該元組進(jìn)棧。 在計(jì)算機(jī)科學(xué)中,閉包 又稱 詞法閉包 或 函數(shù)閉包,是引用了自由變量的函數(shù)。這個(gè)被引用的自由變量將和這個(gè)函數(shù)一同存在,即使已經(jīng)離開(kāi)了創(chuàng)造它的環(huán)境也不例外。閉包被廣泛應(yīng)用于函數(shù)式語(yǔ)言中。 從上面這段話中可以看出閉...
摘要:進(jìn)階細(xì)節(jié)根據(jù)慕課網(wǎng)七月老師視頻整理一切皆對(duì)象對(duì)與來(lái)說(shuō),一切皆對(duì)象,包括函數(shù)。閉包的意義在于返回了一個(gè)現(xiàn)場(chǎng),如果沒(méi)有閉包,很容易被外部的變量所影響。重復(fù)定義可見(jiàn)此時(shí)返回了不再是閉包了。注意該方法的返回值只能是布爾類型,即或。 Python進(jìn)階細(xì)節(jié) 根據(jù)慕課網(wǎng)七月老師視頻整理 一切皆對(duì)象 對(duì)與Python來(lái)說(shuō),一切皆對(duì)象,包括函數(shù)。在其他語(yǔ)言比如c++中,函數(shù)只是一段可執(zhí)行的代碼,只要你獲...
摘要:恩如期來(lái)啦閉包一函數(shù)作為返回值介紹閉包之前,先了解一下函數(shù)作為返回值的情況。例如之前介紹的裝飾器中,就出現(xiàn)了將函數(shù)作為返回值。當(dāng)執(zhí)行時(shí),相當(dāng)于執(zhí)行,且包含。允許使用關(guān)鍵字創(chuàng)造匿名函數(shù)。例如調(diào)用默認(rèn)可以把匿名函數(shù)作為返回值返回,例如 恩~ 如期來(lái)啦閉包~ 一、函數(shù)作為返回值 介紹閉包之前,先了解一下函數(shù)作為返回值的情況。高階函數(shù)除了可以接收函數(shù)作為參數(shù)外,還可以把函數(shù)作為結(jié)果值返回。...
閱讀 1965·2021-09-07 09:59
閱讀 2528·2019-08-29 16:33
閱讀 3702·2019-08-29 16:18
閱讀 2860·2019-08-29 15:30
閱讀 1687·2019-08-29 13:52
閱讀 2050·2019-08-26 18:36
閱讀 545·2019-08-26 12:19
閱讀 708·2019-08-23 15:23