所謂閉包余素,就是將組成函數(shù)的語句和這些語句的執(zhí)行環(huán)境打包在一起時豹休,得到的對象。
下面是閉包的栗子:
#foo.py foo文件
filename = 'foo.py'
def call_func(f): #這個函數(shù)就是說桨吊,要返回調(diào)用那個作為參數(shù)的函數(shù)
return f()
#func.py 另一個叫func的文件
import foo #在這里導(dǎo)入上一個 foo 文件
filename = 'func.py'
def show_filename():
return "filename: %s" %filename
print( foo.call_func(show_filename)) # 返回:filename: func.py
#注意:實(shí)際發(fā)生調(diào)用的位置威根,是在foo.call_func函數(shù)里
這個栗子就是想說明,有兩個文件foo 和 func视乐,其中都有一個同名filename的定義洛搀。在func文件中導(dǎo)入了foo文件后,運(yùn)行foo.call_func(show_filename)炊林,盡管實(shí)際調(diào)用show_filename的位置在foo.py的call_func內(nèi)部姥卢,但是所返回的filename是在func文件中定義的那個卷要。
在下面這個嵌套函數(shù)中看
#enclosed.py
import foo # 導(dǎo)入了poo 文件渣聚,call_func功能: 返回調(diào)用作為參數(shù)的函數(shù)
def wrapper(): #這個wrapper函數(shù)沒有返回值,打印調(diào)用那個call_func函數(shù)
filename = "enclosed.py" #嵌套函數(shù)內(nèi)部也有一個同名filename變量
def show_filename():
return "filename: %s" % filename #執(zhí)行到這句的時候僧叉,就要開始找奕枝,究竟用哪個 filename
print foo.call_func(show_filename) #輸出:filename: enclosed.py
舉這個栗子想說明:閉包將會捕捉內(nèi)層函數(shù)執(zhí)行所需的整個環(huán)境
LEGB法則就是查找順序。
當(dāng)代碼執(zhí)行到filename: %s" % filename時瓶堕,解釋器按照循序查找filename變量:
先在show_filename函數(shù)中找隘道,然后是到wrapper函數(shù)中找(多層嵌套的話,就由內(nèi)而外查找)郎笆,然后到enclosed.py中找谭梗,最后到內(nèi)置模塊。按順序一旦找到就不在到外層去找了宛蚓,如果一直找不到激捏,那就會出現(xiàn)NameError異常。裝飾器&語法糖就是@log
提到閉包的重要特性:封存上下文凄吏,這一特性可以巧妙的被用于現(xiàn)有函數(shù)的包裝远舅,從而為現(xiàn)有函數(shù)更加功能。而這就是裝飾器痕钢。
(原本想找資料加深對裝飾器的理解图柏,但是,現(xiàn)在這個就拿來讀讀吧)
import logging #這個模塊沒具體查任连,是跟記錄日志有關(guān)的
logging.basicConfig(level = logging.INFO)
#通過logging.basicConfig函數(shù)對日志的輸出格式及方式做相關(guān)配置
def add(a,b): #這個add函數(shù)是要被“裝飾的”
return a+b
def checkParams(fn): #通過封存的fn蚤吹,繼續(xù)調(diào)用原始的add進(jìn)行+運(yùn)算。
def wrapper(a,b):
if isinstance(a,(int,float)) and isinstance(b,(int,float)):##檢查參數(shù)a和b是否都為整型或浮點(diǎn)型 isinstance函數(shù)的使用
return fn(a,b) #是則調(diào)用fn(a, b)返回計算結(jié)果
#否則通過logging記錄錯誤信息随抠,并友好退出
logging.warning("variable 'a' and 'b' cannot be added")
return
return wrapper #fn引用add裁着,被封存在閉包的執(zhí)行環(huán)境中返回
#將add函數(shù)對象傳入余佃,fn指向add
#等號左側(cè)的add,指向checkParams的返回值wrapper
add = checkParams(add) #這里一般會被寫成語法糖簡化
add(3,'hello')
誒跨算。爆土。。亂七八糟诸蚕。步势。的
看這個吧https://www.zhihu.com/question/25950466