簡單談?wù)凱ython中的閉包
Python中的閉包
前幾天又有人留言,關(guān)于其中一個(gè)閉包
和re.sub
的使用不太清楚。我在腳本之家搜索了下,發(fā)現(xiàn)沒有寫過閉包相關(guān)的東西,所以決定總結(jié)一下,完善Python的內(nèi)容。
1. 閉包的概念
首先還得從基本概念說起,什么是閉包呢?來看下維基上的解釋:
....
上面提到了兩個(gè)關(guān)鍵的地方: 自由變量 和 函數(shù), 這兩個(gè)關(guān)鍵稍后再說。還是得在贅述下“閉包”的意思,望文知意,可以形象的把它理解為一個(gè)封閉的包裹,這個(gè)包裹就是一個(gè)函數(shù),當(dāng)然還有函數(shù)內(nèi)部對(duì)應(yīng)的邏輯,包裹里面的東西就是自由變量,自由變量可以在隨著包裹到處游蕩。當(dāng)然還得有個(gè)前提,這個(gè)包裹是被創(chuàng)建出來的。
舉個(gè)例子:
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的時(shí)候就產(chǎn)生了一個(gè)閉包——inner_func,并且該閉包持有自由變量——name,因此這也意味著,當(dāng)函數(shù)func的生命周期結(jié)束之后,name這個(gè)變量依然存在,因?yàn)樗婚]包引用了,所以不會(huì)被回收。
另外再說一點(diǎn),閉包并不是Python中特有的概念,所有把函數(shù)做為一等公民的語言均有閉包的概念。不過像Java這樣以class為一等公民的語言中也可以使用閉包,只是它得用類或接口來實(shí)現(xiàn)。
更多概念上的東西可以參考最后的參考鏈接。
2. 為什么使用閉包
基于上面的介紹,不知道讀者有沒有感覺這個(gè)東西和類有點(diǎn)相似,相似點(diǎn)在于他們都提供了對(duì)數(shù)據(jù)的封裝。不同的是閉包本身就是個(gè)方法。和類一樣,我們?cè)诰幊虝r(shí)經(jīng)常會(huì)把通用的東西抽象成類,(當(dāng)然,還有對(duì)現(xiàn)實(shí)世界——業(yè)務(wù)的建模),以復(fù)用通用的功能。閉包也是一樣,當(dāng)我們需要函數(shù)粒度的抽象時(shí),閉包就是一個(gè)很好的選擇。
在這點(diǎn)上閉包可以被理解為一個(gè)只讀的對(duì)象,你可以給他傳遞一個(gè)屬性,但它只能提供給你一個(gè)執(zhí)行的接口。因此在程序中我們經(jīng)常需要這樣的一個(gè)函數(shù)對(duì)象——閉包,來幫我們完成一個(gè)通用的功能,比如后面會(huì)提到的——裝飾器。
3. 使用閉包
第一種場景 ,在python中很重要也很常見的一個(gè)使用場景就是裝飾器,Python為裝飾器提供了一個(gè)很友好的“語法糖”——@,讓我們可以很方便的使用裝飾器,裝飾的原理不做過多闡述,簡言之你在一個(gè)函數(shù)func上加上@decorator_func, 就相當(dāng)于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 # 等價(jià)于 decorator_func(func)
在裝飾器的這個(gè)例子中,閉包(wrapper)持有了外部的func這個(gè)參數(shù),并且能夠接受外部傳過來的參數(shù),接受過來的參數(shù)在原封不動(dòng)的傳給func,并返回執(zhí)行結(jié)果。
這是個(gè)簡單的例子,稍微復(fù)雜點(diǎn)可以有多個(gè)閉包,比如經(jīng)常使用的那個(gè)LRUCache的裝飾器,裝飾器上可以接受參數(shù)@lru_cache(expire=500)這樣。實(shí)現(xiàn)起來就是兩個(gè)閉包的嵌套:
def lru_cache(expire=5): # 默認(rèn)5s超時(shí) 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)常會(huì)問到的面試題。
第二個(gè)場景 ,就是基于閉包的一個(gè)特性——“惰性求值”。這個(gè)應(yīng)用比較常見的是在數(shù)據(jù)庫訪問的時(shí)候,比如說:
# 偽代碼示意 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 # 這時(shí)才執(zhí)行數(shù)據(jù)庫訪問
上面這個(gè)不太恰當(dāng)?shù)睦诱故玖送ㄟ^閉包完成惰性求值的功能,但是上面query返回的結(jié)果并不是函數(shù),而是具有函數(shù)功能的類。有興趣的可以去看看Django的queryset的實(shí)現(xiàn),原理類似。
第三種場景 , 需要對(duì)某個(gè)函數(shù)的參數(shù)提前賦值的情況,當(dāng)然在Python中已經(jīng)有了很好的解決訪問 functools.parial,但是用閉包也能實(shí)現(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") # 當(dāng)然用functools比這個(gè)簡單多了 # 只需要: functools.partial(say, age=15)(name='the5fire')
看起來這又是一個(gè)牽強(qiáng)的例子,不過也算是實(shí)踐了閉包的應(yīng)用。
最后總結(jié)下,閉包這東西理解起來還是很容易的,在Python中的應(yīng)用也很廣泛,這篇文章算是對(duì)閉包的一個(gè)總結(jié),有任何疑問歡迎留言交流。
4. 參考資料
http://stackoverflow.com/questions/4020419/closures-in-python
http://www.shutupandship.com/2012/01/python-closures-explained.html
http://mrevelle.blogspot.com/2006/10/closure-on-closures.html
相關(guān)文章
在python中利用try..except來代替if..else的用法
今天小編就為大家分享一篇在python中利用try..except來代替if..else的用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2019-12-12python實(shí)現(xiàn)H2O中的隨機(jī)森林算法介紹及其項(xiàng)目實(shí)戰(zhàn)
這篇文章主要介紹了python實(shí)現(xiàn)H2O中的隨機(jī)森林算法介紹及其項(xiàng)目實(shí)戰(zhàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08Pygame游戲開發(fā)之太空射擊實(shí)戰(zhàn)盾牌篇
相信大多數(shù)8090后都玩過太空射擊游戲,在過去游戲不多的年代太空射擊自然屬于經(jīng)典好玩的一款了,今天我們來自己動(dòng)手實(shí)現(xiàn)它,在編寫學(xué)習(xí)中回顧過往展望未來,在本課中,我們將為玩家添加一個(gè)盾牌以及一個(gè)用于顯示盾牌等級(jí)的欄2022-08-08django中模板繼承與ModelForm實(shí)例詳解
ModelForm類是form是組件中Form的一個(gè)子類,所以也是處理表單的,下面這篇文章主要給大家介紹了關(guān)于django中模板繼承與ModelForm的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-04-04Python實(shí)現(xiàn)為圖片添加水印的示例詳解
這篇文章主要介紹了如何通過Python3實(shí)現(xiàn)添加水印,這樣發(fā)朋友圈,圖片再也不怕被盜了?。。∥闹械氖纠a簡潔易懂,需要的可以參考一下2022-02-02python3的UnicodeDecodeError解決方法
這篇文章主要介紹了python3的UnicodeDecodeError解決方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12