如何理解Python中的閉包Closure

如何理解Python中的閉包Closure,針對這個(gè)問題,這篇文章詳細(xì)介紹了相對應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問題的小伙伴找到更簡單易行的方法。

創(chuàng)新互聯(lián)建站-專業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性價(jià)比東方網(wǎng)站開發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫,直接使用。一站式東方網(wǎng)站制作公司更省心,省錢,快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋東方地區(qū)。費(fèi)用合理售后完善,十多年實(shí)體公司更值得信賴。

Python中的閉包不是一個(gè)一說就能明白的概念,但是隨著你往學(xué)習(xí)的深入,無論如何你都需要去了解這么一個(gè)東西。

閉包的概念

我們嘗試從概念上去理解一下閉包。

在一些語言中,在函數(shù)中可以(嵌套)定義另一個(gè)函數(shù)時(shí),如果內(nèi)部的函數(shù)引用了外部的函數(shù)的變量,則可能產(chǎn)生閉包。閉包可以用來在一個(gè)函數(shù)與一組“私有”變量之間創(chuàng)建關(guān)聯(lián)關(guān)系。在給定函數(shù)被多次調(diào)用的過程中,這些私有變量能夠保持其持久性。

—— 維基百科)

用比較容易懂的人話說,就是當(dāng)某個(gè)函數(shù)被當(dāng)成對象返回時(shí),夾帶了外部變量,就形成了一個(gè)閉包。看例子。

def make_printer(msg):     def printer():         print msg  # 夾帶私貨(外部變量)     return printer  # 返回的是函數(shù),帶私貨的函數(shù)  printer = make_printer('Foo!') printer()

支持將函數(shù)當(dāng)成對象使用的編程語言,一般都支持閉包。比如Python, JavaScript。

如何理解閉包

閉包存在有什么意義呢?為什么需要閉包?

我個(gè)人認(rèn)為,閉包存在的意義就是它夾帶了外部變量(私貨),如果它不夾帶私貨,它和普通的函數(shù)就沒有任何區(qū)別。同一個(gè)的函數(shù)夾帶了不同的私貨,就實(shí)現(xiàn)了不同的功能。其實(shí)你也可以這么理解,閉包和面向接口編程的概念很像,可以把閉包理解成輕量級的接口封裝。

接口定義了一套對方法簽名的約束規(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) # <a>Hello</a>  add_tag = tag('b') print add_tag(content) # <b>Hello</b>

在這個(gè)例子里,我們想要一個(gè)給content加tag的功能,但是具體的tag_name是什么樣子的要根據(jù)實(shí)際需求來定,對外部調(diào)用的接口已經(jīng)確定,就是add_tag(content)。如果按照面向接口方式實(shí)現(xiàn),我們會先把a(bǔ)dd_tag寫成接口,指定其參數(shù)和返回類型,然后分別去實(shí)現(xiàn)a和b的add_tag。

但是在閉包的概念中,add_tag就是一個(gè)函數(shù),它需要tag_name和content兩個(gè)參數(shù),只不過tag_name這個(gè)參數(shù)是打包帶走的。所以一開始時(shí)就可以告訴我怎么打包,然后帶走就行。

上面的例子不太生動,其實(shí)在我們生活和工作中,閉包的概念也很常見。比如說手機(jī)撥號,你只關(guān)心電話打給誰,而不會去糾結(jié)每個(gè)品牌的手機(jī)是怎么實(shí)現(xiàn)的,用到了哪些模塊。再比如去餐館吃飯,你只要付錢就可以享受到服務(wù),你并不知道那桌飯菜用了多少地溝油。這些都可以看成閉包,返回來的是一些功能或者服務(wù)(打電話,用餐),但是這些功能使用了外部變量(天線,地溝油等等)。

你也可以把一個(gè)類實(shí)例看成閉包,當(dāng)你在構(gòu)造這個(gè)類時(shí),使用了不同的參數(shù),這些參數(shù)就是閉包里的包,這個(gè)類對外提供的方法就是閉包的功能。但是類遠(yuǎn)遠(yuǎn)大于閉包,因?yàn)殚]包只是一個(gè)可以執(zhí)行的函數(shù),但是類實(shí)例則有可能提供很多方法。

何時(shí)使用閉包

其實(shí)閉包在Python中很常見,只不過你沒特別注意這就是一個(gè)閉包。比如Python中的裝飾器Decorator,假如你需要寫一個(gè)帶參數(shù)的裝飾器,那么一般都會生成閉包。

為什么?因?yàn)镻ython的裝飾器是一個(gè)固定的函數(shù)接口形式。它要求你的裝飾器函數(shù)(或裝飾器類)必須接受一個(gè)函數(shù)并返回一個(gè)函數(shù):

# how to define def wrapper(func1):  # 接受一個(gè)callable對象     return func2  # 返回一個(gè)對象,一般為函數(shù)      # how to use def target_func(args): # 目標(biāo)函數(shù)     pass  # 調(diào)用方式一,直接包裹 result = wrapper(target_func)(args)  # 調(diào)用方式二,使用@語法,等同于方式一 @wrapper def target_func(args):     pass  result = target_func()

那么如果你的裝飾器如果帶參數(shù)呢?那么你就需要在原來的裝飾器上再包一層,用于接收這些參數(shù)。這些參數(shù)(私貨)傳遞到內(nèi)層的裝飾器里后,閉包就形成了。所以說當(dāng)你的裝飾器需要自定義參數(shù)時(shí),一般都會形成閉包。(類裝飾器例外)

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)  # 不用@的寫法如下 # hello = html_tag('b')(hello) # html_tag('b') 是一個(gè)閉包,它接受一個(gè)函數(shù),并返回一個(gè)函數(shù)  print hello()  # <b>Hello Toby!</b> print hello('world')  # <b>Hello world!</b>

關(guān)于裝飾器的更深入剖析,可以看我寫的另外一篇博客。

再深入一點(diǎn)

其實(shí)也不必太深入,理解這上面的概念,很多看起來頭疼的代碼也不過如此。

下面讓我們來了解一下閉包的包到底長什么樣子。其實(shí)閉包函數(shù)相對與普通函數(shù)會多出一個(gè)__closure__的屬性,里面定義了一個(gè)元組用于存放所有的cell對象,每個(gè)cell對象一一保存了這個(gè)閉包中所有的外部變量。

>>> def make_printer(msg1, msg2):     def printer():         print msg1, msg2     return printer >>> printer = make_printer('Foo', 'Bar')  # 形成閉包  >>> printer.__closure__   # 返回cell元組 (<cell at 0x03A10930: str object at 0x039DA218>, <cell at 0x03A10910: str object at 0x039DA488>)  >>> printer.__closure__[0].cell_contents  # ***個(gè)外部變量 'Foo' >>> printer.__closure__[1].cell_contents  # 第二個(gè)外部變量 'Bar'

原理就是這么簡單。

關(guān)于如何理解Python中的閉包Closure問題的解答就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道了解更多相關(guān)知識。

本文名稱:如何理解Python中的閉包Closure
鏈接地址:http://bm7419.com/article26/jddscg.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供靜態(tài)網(wǎng)站、品牌網(wǎng)站設(shè)計(jì)App開發(fā)、電子商務(wù)、用戶體驗(yàn)、移動網(wǎng)站建設(shè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

商城網(wǎng)站建設(shè)