1.《python核心编程》的这段程序怎么理解?对于这个函数,书上说两个wraaped是闭包,但是不知道谁是自由变量。
2.代码:
from time import time
def logged(when):
def log(f, *args, **kwargs):
print '''Called:
function: %s
args: %r
kwargs %r''' % (f, args, kwargs)
def pre_logged(f):
def wraper(*args, **kwargs):
log(f, *args, **kwargs)
return f(*args, **kwargs)
return wraper
def post_logged(f):
def wrapped(*args, **kwargs):
now = time()
try:
return f(*args, **kwargs)
finally:
log(f, *args, **kwargs)
print "time delta: %s" % (time()-now)
return wrapped
try:
return {"pre": pre_logged, "post": post_logged}[when]
except KeyError, e:
raise ValueError(e), 'must bre "pre" or "post"'
@logged("post")
def hello(name):
print "Hello, ", name
hello('World!')
所謂閉包,就是指在函數中定義的函數。其實嚴格的說,你上面的函數logged下面的所有函數定義都是閉包
關於閉包:
當一個內嵌函數引用外部作用域的變量,我們就會得到一個閉包。建立一個閉包必須同時滿足以下幾點:
1、必須要有一個內嵌函數。題目中外部函數是pre_logged,post_logged,對應的內嵌函數為wraper,wrapped。
2、內嵌函數中必須要引用外部函數的變數。題目中是引用來外部的args, *kwargs參數。
3、外部函數回傳值必須是內嵌函數。題目中是用return wraper, return wrapped加以返回。
關於裝飾器:
簡單的說裝飾器就是修改其他函數功能的函數。題中logged就是一個裝飾器,他用來裝飾你定義的hello函數。
裝飾器參數你傳入了“post”,根據:return {"pre": pre_logged, "post": post_logged}[when]
調用了post_logged函數,post_logged的功能是打印花費的時間(time()- now)
關於裝飾器的詳細解釋,可以參考《Python進階》:
裝飾器詳解
這整本書都翻譯的不錯,推薦樓主看看。