真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

python3函數(shù)閉包 python中閉包

python函數(shù)的閉包怎么理解

1. 閉包的概念

創(chuàng)新互聯(lián)公司是一家專注網(wǎng)站建設(shè)、網(wǎng)絡(luò)營銷策劃、微信小程序、電子商務(wù)建設(shè)、網(wǎng)絡(luò)推廣、移動互聯(lián)開發(fā)、研究、服務(wù)為一體的技術(shù)型公司。公司成立10年以來,已經(jīng)為1000+成都航空箱各業(yè)的企業(yè)公司提供互聯(lián)網(wǎng)服務(wù)?,F(xiàn)在,服務(wù)的1000+客戶與我們一路同行,見證我們的成長;未來,我們一起分享成功的喜悅。

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

復(fù)制代碼代碼如下:

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

....

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

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

舉個例子:

復(fù)制代碼代碼如下:

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,因此這也意味著,當(dāng)函數(shù)func的生命周期結(jié)束之后,name這個變量依然存在,因為它被閉包引用了,所以不會被回收。

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

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

2. 為什么使用閉包

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

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

3. 使用閉包

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

復(fù)制代碼代碼如下:

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)起來就是兩個閉包的嵌套:

復(fù)制代碼代碼如下:

def lru_cache(expire=5):

# 默認(rèn)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ù)庫訪問的時候,比如說:

復(fù)制代碼代碼如下:

# 偽代碼示意

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ù)庫訪問

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

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

復(fù)制代碼代碼如下:

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比這個簡單多了

# 只需要: functools.partial(say, age=15)(name='the5fire')

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

Python基礎(chǔ)之閉包

一.閉包的定義:

在一個函數(shù)的內(nèi)部,再定義一個函數(shù)(內(nèi)部函數(shù))。這個內(nèi)部函數(shù)引用了外部函數(shù)的變量,并且外部函數(shù)返回這個內(nèi)部函數(shù), 我們把這個使用外部函數(shù)變量的內(nèi)部函數(shù)稱為 閉包 。

簡而言之, 閉包就是能夠讀取外部函數(shù)內(nèi)的變量的函數(shù)。

例如:

形成閉包的兩個條件:

二.閉包的用途

① 可以讀取函數(shù)內(nèi)部的變量

② 將一些變量的值始終保存到內(nèi)存中

1.讀取函數(shù)內(nèi)部的變量

在一般情況下,在函數(shù)外部我們是不能訪問到函數(shù)內(nèi)部的變量的。但是, 有時想要在函數(shù)外部能夠訪問到函數(shù)內(nèi)部的變量,那么就可以使用閉包。

例如:

上面的代碼可以看出,print(a)會拋異常NameError: name 'a' is not defined。在函數(shù)f1的外面無法訪問它的變量的。

在函數(shù)f1里面定義一個閉包函數(shù)就可以訪問到了

例如:

2.將一些變量的值始終保存到內(nèi)存中

運行結(jié)果:

通過上面的輸出結(jié)果可以看出閉包保存了外部函數(shù)內(nèi)的變量n1的值1,每次執(zhí)行閉包都是在n1 = 1 基礎(chǔ)上進行計算的。

三.閉包的缺點

1. 由于閉包會使得函數(shù)中的變量都被保存在內(nèi)存中,會增加 內(nèi)存消耗 ,所以不能濫用閉包,否則會造成程序的性能問題,可能導(dǎo)致內(nèi)存泄露

2. 閉包無法改變外部函數(shù)局部變量指向的內(nèi)存地址

3. 返回閉包時,返回函數(shù)不要引用任何循環(huán)變量,或者后續(xù)會發(fā)生變化的變量

四.判斷一個函數(shù)是否是閉包

判斷一個函數(shù)是不是閉包,可以查看它的 closure 屬性。如果該函數(shù)是閉包,查看該屬性將會返回一個cell對象組成的tuple。如果我們分別對每個cell對象查看其cell_contents屬性,返回的內(nèi)容就是閉包引用的自由變量的值。

運行結(jié)果:

閉包的__closure__方法,可以展示出閉包儲存了外部函數(shù)的兩個變量,cell的內(nèi)存地址是什么,在cell里面儲存的對象類型是int,這個int儲存的內(nèi)存地址是什么。

閉包的__closure__方法,可以查看每個cell對象的內(nèi)容

運行結(jié)果:

cell_contents解釋了局部變量在脫離函數(shù)后仍然可以在函數(shù)之外被訪問的原因,因為變量被存儲在cell_contents中了。

python什么是閉包 閉包的作用域

在函數(shù)中可以定義另一個函數(shù)時,如果內(nèi)部的函數(shù)引用了外部的函數(shù)的變量,則可能產(chǎn)生閉包。

閉包可以用來在一個函數(shù)與一組私有變量之間創(chuàng)建關(guān)聯(lián)關(guān)系。

在給定函數(shù)被多次調(diào)用的過程中,這些私有變量能夠保持其持久性。

形成閉包的三個條件

必須有一個內(nèi)嵌函數(shù)—這對應(yīng)函數(shù)之間的嵌套;

內(nèi)嵌函數(shù)必須引用一個定義在閉合范圍內(nèi)的變量—內(nèi)部函數(shù)引用外部變量;

外部函數(shù)必須返回內(nèi)嵌函數(shù)—必須返回內(nèi)部函數(shù)。

換句話來說:閉包的概念很簡單,一個可以引用在函數(shù)閉合范圍內(nèi)變量的函數(shù),即內(nèi)部函數(shù),只有那個內(nèi)部函數(shù)才有所謂的__closure__屬性。

閉包的原理

形成閉包之后,閉包函數(shù)會獲得一個非空的_Closure_屬性,這個屬性是一個元組。

組里面的對象為cell對象,而訪問cell對象的cell_contents屬性則可以得到閉包變量的當(dāng)前值。

而隨著閉包的繼續(xù)調(diào)用,變量會進行再次更新。由此可見,一般形成閉包之后,Python確定會將_closure_和閉包函數(shù)綁定作為儲存閉包變量的場所。

閉包的好處是什么?

其實,閉包并不是必須的。

沒有閉包的話,Python的功能不會受到任何影響;但有了閉包之后,可以提供一種額外的解決方案。


當(dāng)前題目:python3函數(shù)閉包 python中閉包
網(wǎng)站鏈接:http://weahome.cn/article/dogshjc.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部