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