0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

用最簡單的語言解釋Python的閉包是什么?

馬哥Linux運維 ? 來源:YXQ ? 2019-03-21 16:33 ? 次閱讀

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

閉包的概念

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

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

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

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

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

如何理解閉包

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

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

接口定義了一套對方法簽名的約束規(guī)則。

def tag(tag_name):  def add_tag(content):    return "<{0}>{1}".format(tag_name, content)  return add_tag  content = 'Hello'add_tag = tag('a')print add_tag(content)# Helloadd_tag = tag('b')print add_tag(content)# Hello

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

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

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

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

何時使用閉包

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

為什么?因為Python的裝飾器是一個固定的函數(shù)接口。它要求你的裝飾器函數(shù)(或裝飾器類)必須返回這樣一種接口,接受一個函數(shù)并返回一個函數(shù):

# how to definedef wrapper(func1): # 必須接受一個且僅一個函數(shù)作為參數(shù)  return func2 # 返回一個且僅一個callable對象,一般為函數(shù)# how to usedef target_func(args): # 目標(biāo)函數(shù)  pass# 調(diào)用方式一,直接包裹result = wrapper(target_func)(args)# 調(diào)用方式二,使用@語法,等同于方式一@wrapperdef target_func(args):  passresult = target_func()

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

def html_tags(tag_name):  def wrapper_(func):    def wrapper(*args, **kwargs):      content = func(*args, **kwargs)      return "<{tag}>{content}".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') 是一個閉包,它接受一個函數(shù),并返回一個函數(shù)print hello() # Hello Toby!print hello('world') # Hello world!

再深入一點

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

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

>>> 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 # 第一個外部變量'Foo'>>> printer.__closure__[1].cell_contents # 第二個外部變量'Bar'

原理就是這么簡單。

參考鏈接

https://www.the5fire.com/closure-in-python.html

http://stackoverflow.com/questions/4020419/why-arent-python-nested-functions-called-closures

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • python
    +關(guān)注

    關(guān)注

    56

    文章

    4801

    瀏覽量

    84859
  • 閉包
    +關(guān)注

    關(guān)注

    0

    文章

    4

    瀏覽量

    2070

原文標(biāo)題:用最簡單的語言解釋Python的閉包是什么?

文章出處:【微信號:magedu-Linux,微信公眾號:馬哥Linux運維】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    對比Python與Java編程語言

    使得編寫代碼更加靈活,但也可能導(dǎo)致運行時錯誤。 Java 語法相對冗長,需要顯式聲明變量類型,增加了代碼的可讀性和安全性。 靜態(tài)類型系統(tǒng)在編譯時進行類型檢查,減少了運行時錯誤。 二、性能 Python 解釋語言,運行速度通常比
    的頭像 發(fā)表于 11-15 09:31 ?380次閱讀

    單片機STM32可以Python寫嗎?可以的開發(fā)板有哪些?

    近年來,隨著嵌入式技術(shù)的發(fā)展,Python語言逐漸被引入到單片機開發(fā)中,尤其是一些高性能的單片機上。這一趨勢給開發(fā)者帶來了極大的便利,尤其是在快速原型設(shè)計和實驗中。本文將詳細探討STM32單片機是否
    的頭像 發(fā)表于 09-05 08:00 ?4004次閱讀
    單片機STM32可以<b class='flag-5'>用</b><b class='flag-5'>Python</b>寫嗎?可以的開發(fā)板有哪些?

    使用Python進行Ping測試

    )請求,然后等待目標(biāo)主機返回響應(yīng),從而測量網(wǎng)絡(luò)的延遲和丟情況。隨著Python編程語言的廣泛應(yīng)用,越來越多的網(wǎng)絡(luò)工程師開始使用
    的頭像 發(fā)表于 08-12 17:56 ?561次閱讀
    使用<b class='flag-5'>Python</b>進行Ping測試

    pytorch和python的關(guān)系是什么

    ,PyTorch已經(jīng)成為了一個非常受歡迎的框架。本文將介紹PyTorch和Python之間的關(guān)系,以及它們在深度學(xué)習(xí)領(lǐng)域的應(yīng)用。 Python簡介 Python是一種高級、解釋型、通用
    的頭像 發(fā)表于 08-01 15:27 ?2124次閱讀

    Python建模算法與應(yīng)用

    Python作為一種功能強大、免費、開源且面向?qū)ο蟮木幊?b class='flag-5'>語言,在科學(xué)計算、數(shù)學(xué)建模、數(shù)據(jù)分析等領(lǐng)域展現(xiàn)出了卓越的性能。其簡潔的語法、對動態(tài)輸入的支持以及解釋語言的本質(zhì),使得
    的頭像 發(fā)表于 07-24 10:41 ?604次閱讀

    python寫驗證環(huán)境cocotb

    本文介紹了cocotb的安裝、python tb文件的寫法、xrun仿真cocotb的腳本等,我們來看看體驗如何。
    的頭像 發(fā)表于 07-24 09:38 ?582次閱讀
    <b class='flag-5'>用</b><b class='flag-5'>python</b>寫驗證環(huán)境cocotb

    Python語言基礎(chǔ)2

    電子發(fā)燒友網(wǎng)站提供《Python語言基礎(chǔ)2.rar》資料免費下載
    發(fā)表于 07-17 17:23 ?1次下載

    opencv-python和opencv一樣嗎

    是OpenCV的一個Python語言接口,它允許開發(fā)者使用Python語言來調(diào)用OpenCV庫的功能。 雖然OpenCV和OpenCV-Python
    的頭像 發(fā)表于 07-16 10:38 ?1299次閱讀

    pycharm進行python爬蟲的步驟

    PyCharm,然后點擊“Create New Project”創(chuàng)建一個新的Python項目。選擇一個合適的項目名稱和位置,然后選擇Python解釋器。如果您還沒有安裝Py
    的頭像 發(fā)表于 07-11 10:11 ?901次閱讀

    tensorflow簡單的模型訓(xùn)練

    TensorFlow開始,然后介紹如何構(gòu)建和訓(xùn)練一個簡單的神經(jīng)網(wǎng)絡(luò)模型。 1. 安裝TensorFlow 首先,我們需要安裝TensorFlow。TensorFlow支持多種編程語言,包括Python、C++
    的頭像 發(fā)表于 07-05 09:38 ?732次閱讀

    使用Python進行自然語言處理

    在探討使用Python進行自然語言處理(NLP)的廣闊領(lǐng)域時,我們首先需要理解NLP的基本概念、其重要性、Python在NLP中的優(yōu)勢,以及如何通過Python實現(xiàn)一些基礎(chǔ)的NLP任務(wù)
    的頭像 發(fā)表于 07-04 14:40 ?476次閱讀

    安裝依賴的Python軟件時報錯如何解決?

    安裝依賴的 Python 軟件,出現(xiàn)了如下錯誤,請問如何解決,謝謝! Command \"E:/msys32/mingw32/bin/python2.7.exe E:/msys32
    發(fā)表于 06-26 07:32

    華為云開發(fā)者桌面全新發(fā)布 CodeArts IDE for Python,極致優(yōu)雅云原生開發(fā)體驗

    Python 編碼體驗。 Python 是一種編程語言,廣泛用于 Web 應(yīng)用程序、軟件開發(fā)、數(shù)據(jù)科學(xué)和機器學(xué)習(xí) (ML)。Python 以其優(yōu)雅的語法、動態(tài)
    的頭像 發(fā)表于 05-10 00:27 ?1263次閱讀
    華為云開發(fā)者桌面全新發(fā)布 CodeArts IDE for <b class='flag-5'>Python</b>,極致優(yōu)雅云原生開發(fā)體驗

    fpga的是什么編程語言 fpga什么語言開發(fā)

    fpga的是什么編程語言 FPGA(現(xiàn)場可編程邏輯門陣列)主要使用的編程語言是硬件描述語言(HDL)。在眾多的HDL中,Verilog HDL和VHDL是最常用的兩種。 Verilo
    的頭像 發(fā)表于 03-14 17:09 ?3572次閱讀

    c語言,c++,java,python區(qū)別

    C語言、C++、Java和Python是四種常見的編程語言,各有優(yōu)點和特點。 C語言: C語言是一種面向過程的編程
    的頭像 發(fā)表于 02-05 14:11 ?2501次閱讀