亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

簡單談?wù)凱ython中的閉包

 更新時間:2016年11月30日 08:40:29   作者:無落塵埃  
一般來說閉包這個概念在很多語言中都有涉及,簡單說,閉包就是根據(jù)不同的配置信息得到不同的結(jié)果,下面我們來專門講下在Python中的閉包

Python中的閉包

前幾天又有人留言,關(guān)于其中一個閉包re.sub的使用不太清楚。我在腳本之家搜索了下,發(fā)現(xiàn)沒有寫過閉包相關(guān)的東西,所以決定總結(jié)一下,完善Python的內(nèi)容。

1. 閉包的概念

首先還得從基本概念說起,什么是閉包呢?來看下維基上的解釋:

復(fù)制代碼 代碼如下:
在計算機科學(xué)中,閉包(Closure)是詞法閉包(Lexical Closure)的簡稱,是引用了自由變量的函數(shù)。這個被引用的自由變量將和這個函數(shù)一同存在,即使已經(jīng)離開了創(chuàng)造它的環(huán)境也不例外。所以,有另一種說法認為閉包是由函數(shù)和與其相關(guān)的引用環(huán)境組合而成的實體。閉包在運行時可以有多個實例,不同的引用環(huán)境和相同的函數(shù)組合可以產(chǎn)生不同的實例。
....

上面提到了兩個關(guān)鍵的地方: 自由變量 和 函數(shù), 這兩個關(guān)鍵稍后再說。還是得在贅述下“閉包”的意思,望文知意,可以形象的把它理解為一個封閉的包裹,這個包裹就是一個函數(shù),當然還有函數(shù)內(nèi)部對應(yīng)的邏輯,包裹里面的東西就是自由變量,自由變量可以在隨著包裹到處游蕩。當然還得有個前提,這個包裹是被創(chuàng)建出來的。

在通過Python的語言介紹一下,一個閉包就是你調(diào)用了一個函數(shù)A,這個函數(shù)A返回了一個函數(shù)B給你。這個返回的函數(shù)B就叫做閉包。你在調(diào)用函數(shù)A的時候傳遞的參數(shù)就是自由變量。

舉個例子:

def func(name):
 def inner_func(age):
  print 'name:', name, 'age:', age
 return inner_func

bb = func('the5fire')
bb(26) # >>> name: the5fire age: 26

這里面調(diào)用func的時候就產(chǎn)生了一個閉包——inner_func,并且該閉包持有自由變量——name,因此這也意味著,當函數(shù)func的生命周期結(jié)束之后,name這個變量依然存在,因為它被閉包引用了,所以不會被回收。

另外再說一點,閉包并不是Python中特有的概念,所有把函數(shù)做為一等公民的語言均有閉包的概念。不過像Java這樣以class為一等公民的語言中也可以使用閉包,只是它得用類或接口來實現(xiàn)。

更多概念上的東西可以參考最后的參考鏈接。

2. 為什么使用閉包

基于上面的介紹,不知道讀者有沒有感覺這個東西和類有點相似,相似點在于他們都提供了對數(shù)據(jù)的封裝。不同的是閉包本身就是個方法。和類一樣,我們在編程時經(jīng)常會把通用的東西抽象成類,(當然,還有對現(xiàn)實世界——業(yè)務(wù)的建模),以復(fù)用通用的功能。閉包也是一樣,當我們需要函數(shù)粒度的抽象時,閉包就是一個很好的選擇。

在這點上閉包可以被理解為一個只讀的對象,你可以給他傳遞一個屬性,但它只能提供給你一個執(zhí)行的接口。因此在程序中我們經(jīng)常需要這樣的一個函數(shù)對象——閉包,來幫我們完成一個通用的功能,比如后面會提到的——裝飾器。

3. 使用閉包

第一種場景 ,在python中很重要也很常見的一個使用場景就是裝飾器,Python為裝飾器提供了一個很友好的“語法糖”——@,讓我們可以很方便的使用裝飾器,裝飾的原理不做過多闡述,簡言之你在一個函數(shù)func上加上@decorator_func, 就相當于decorator_func(func):

def decorator_func(func):
 def wrapper(*args, **kwargs):
  return func(*args, **kwargs)
 return wrapper

@decorator_func
def func(name):
 print 'my name is', name

# 等價于
decorator_func(func)

在裝飾器的這個例子中,閉包(wrapper)持有了外部的func這個參數(shù),并且能夠接受外部傳過來的參數(shù),接受過來的參數(shù)在原封不動的傳給func,并返回執(zhí)行結(jié)果。

這是個簡單的例子,稍微復(fù)雜點可以有多個閉包,比如經(jīng)常使用的那個LRUCache的裝飾器,裝飾器上可以接受參數(shù)@lru_cache(expire=500)這樣。實現(xiàn)起來就是兩個閉包的嵌套:

def lru_cache(expire=5):
 # 默認5s超時
 def func_wrapper(func):
  def inner(*args, **kwargs):
   # cache 處理 bala bala bala
   return func(*args, **kwargs)
  return inner
 return func_wrapper

@lru_cache(expire=10*60)
def get(request, pk)
 # 省略具體代碼
 return response()

不太懂閉包的同學(xué)一定得能夠理解上述代碼,這是我們之前面試經(jīng)常會問到的面試題。

第二個場景 ,就是基于閉包的一個特性——“惰性求值”。這個應(yīng)用比較常見的是在數(shù)據(jù)庫訪問的時候,比如說:

# 偽代碼示意

class QuerySet(object):
 def __init__(self, sql):
  self.sql = sql
  self.db = Mysql.connect().corsor() # 偽代碼

 def __call__(self):
  return db.execute(self.sql)

def query(sql):
 return QuerySet(sql)

result = query("select name from user_app")
if time > now:
 print result # 這時才執(zhí)行數(shù)據(jù)庫訪問

上面這個不太恰當?shù)睦诱故玖送ㄟ^閉包完成惰性求值的功能,但是上面query返回的結(jié)果并不是函數(shù),而是具有函數(shù)功能的類。有興趣的可以去看看Django的queryset的實現(xiàn),原理類似。

第三種場景 , 需要對某個函數(shù)的參數(shù)提前賦值的情況,當然在Python中已經(jīng)有了很好的解決訪問 functools.parial,但是用閉包也能實現(xiàn)。

def partial(**outer_kwargs):
 def wrapper(func):
  def inner(*args, **kwargs):
   for k, v in outer_kwargs.items():
    kwargs[k] = v
   return func(*args, **kwargs)
  return inner
 return wrapper

@partial(age=15)
def say(name=None, age=None):
 print name, age

say(name="the5fire")
# 當然用functools比這個簡單多了
# 只需要: functools.partial(say, age=15)(name='the5fire')

看起來這又是一個牽強的例子,不過也算是實踐了閉包的應(yīng)用。

最后總結(jié)下,閉包這東西理解起來還是很容易的,在Python中的應(yīng)用也很廣泛,這篇文章算是對閉包的一個總結(jié),有任何疑問歡迎留言交流。

4. 參考資料

維基百科-閉包

http://stackoverflow.com/questions/4020419/closures-in-python

http://www.shutupandship.com/2012/01/python-closures-explained.html

http://stackoverflow.com/questions/141642/what-limitations-have-closures-in-python-compared-to-language-x-closures

http://mrevelle.blogspot.com/2006/10/closure-on-closures.html

相關(guān)文章

  • 基于Python開發(fā)chrome插件的方法分析

    基于Python開發(fā)chrome插件的方法分析

    這篇文章主要介紹了基于Python開發(fā)chrome插件的方法,結(jié)合實例形式分析了Python實現(xiàn)chrome瀏覽器插件相關(guān)操作技巧,需要的朋友可以參考下
    2018-07-07
  • 在python中利用try..except來代替if..else的用法

    在python中利用try..except來代替if..else的用法

    今天小編就為大家分享一篇在python中利用try..except來代替if..else的用法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-12-12
  • python實現(xiàn)H2O中的隨機森林算法介紹及其項目實戰(zhàn)

    python實現(xiàn)H2O中的隨機森林算法介紹及其項目實戰(zhàn)

    這篇文章主要介紹了python實現(xiàn)H2O中的隨機森林算法介紹及其項目實戰(zhàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-08-08
  • 寶塔面板成功部署Django項目流程(圖文)

    寶塔面板成功部署Django項目流程(圖文)

    這篇文章主要介紹了寶塔面板成功部署Django項目流程,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-06-06
  • Pygame游戲開發(fā)之太空射擊實戰(zhàn)盾牌篇

    Pygame游戲開發(fā)之太空射擊實戰(zhàn)盾牌篇

    相信大多數(shù)8090后都玩過太空射擊游戲,在過去游戲不多的年代太空射擊自然屬于經(jīng)典好玩的一款了,今天我們來自己動手實現(xiàn)它,在編寫學(xué)習(xí)中回顧過往展望未來,在本課中,我們將為玩家添加一個盾牌以及一個用于顯示盾牌等級的欄
    2022-08-08
  • django中模板繼承與ModelForm實例詳解

    django中模板繼承與ModelForm實例詳解

    ModelForm類是form是組件中Form的一個子類,所以也是處理表單的,下面這篇文章主要給大家介紹了關(guān)于django中模板繼承與ModelForm的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-04-04
  • 利用Python抓取行政區(qū)劃碼的方法

    利用Python抓取行政區(qū)劃碼的方法

    做項目的時候會需要用到各個行政區(qū)劃的代碼,最近就碰巧遇到有這個需求,于是就上網(wǎng)搜了一下,測試后分享給大家,這篇文章就給大家分享了利用Python抓取行政區(qū)劃碼的示例代碼,有需要的朋友們可以參考借鑒,下面跟著小編一起去學(xué)習(xí)學(xué)習(xí)吧。
    2016-11-11
  • 利用Python編寫一個簡單的緩存系統(tǒng)

    利用Python編寫一個簡單的緩存系統(tǒng)

    今天來做一個最簡單的例子,利用寫一個最簡單的緩存系統(tǒng),以key``value的方式保持數(shù)據(jù),并且需要將內(nèi)容中的數(shù)據(jù)落地到文件,以便下次啟動的時候,將文件的內(nèi)容加載進內(nèi)存中來,感興趣的可以了解一下
    2023-04-04
  • Python實現(xiàn)為圖片添加水印的示例詳解

    Python實現(xiàn)為圖片添加水印的示例詳解

    這篇文章主要介紹了如何通過Python3實現(xiàn)添加水印,這樣發(fā)朋友圈,圖片再也不怕被盜了!??!文中的示例代碼簡潔易懂,需要的可以參考一下
    2022-02-02
  • python3的UnicodeDecodeError解決方法

    python3的UnicodeDecodeError解決方法

    這篇文章主要介紹了python3的UnicodeDecodeError解決方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-12-12

最新評論