Python如何創(chuàng)建裝飾器時(shí)保留函數(shù)元信息
問(wèn)題
你寫(xiě)了一個(gè)裝飾器作用在某個(gè)函數(shù)上,但是這個(gè)函數(shù)的重要的元信息比如名字、文檔字符串、注解和參數(shù)簽名都丟失了。
解決方案
任何時(shí)候你定義裝飾器的時(shí)候,都應(yīng)該使用 functools 庫(kù)中的 @wraps 裝飾器來(lái)注解底層包裝函數(shù)。例如:
import timefrom functools import wrapsdef timethis(func): ’’’ Decorator that reports the execution time. ’’’ @wraps(func) def wrapper(*args, **kwargs): start = time.time() result = func(*args, **kwargs) end = time.time() print(func.__name__, end-start) return result return wrapper
下面我們使用這個(gè)被包裝后的函數(shù)并檢查它的元信息:
>>> @timethis... def countdown(n):... ’’’... Counts down... ’’’... while n > 0:... n -= 1...>>> countdown(100000)countdown 0.008917808532714844>>> countdown.__name__’countdown’>>> countdown.__doc__’ntCounts downnt’>>> countdown.__annotations__{’n’: <class ’int’>}>>>
討論
在編寫(xiě)裝飾器的時(shí)候復(fù)制元信息是一個(gè)非常重要的部分。如果你忘記了使用 @wraps , 那么你會(huì)發(fā)現(xiàn)被裝飾函數(shù)丟失了所有有用的信息。比如如果忽略 @wraps 后的效果是下面這樣的:
>>> countdown.__name__’wrapper’>>> countdown.__doc__>>> countdown.__annotations__{}>>>
@wraps 有一個(gè)重要特征是它能讓你通過(guò)屬性 __wrapped__ 直接訪(fǎng)問(wèn)被包裝函數(shù)。例如:
>>> countdown.__wrapped__(100000)>>>
__wrapped__ 屬性還能讓被裝飾函數(shù)正確暴露底層的參數(shù)簽名信息。例如:
>>> from inspect import signature>>> print(signature(countdown))(n:int)>>>
一個(gè)很普遍的問(wèn)題是怎樣讓裝飾器去直接復(fù)制原始函數(shù)的參數(shù)簽名信息, 如果想自己手動(dòng)實(shí)現(xiàn)的話(huà)需要做大量的工作,最好就簡(jiǎn)單的使用 @wraps 裝飾器。 通過(guò)底層的 __wrapped__ 屬性訪(fǎng)問(wèn)到函數(shù)簽名信息。
以上就是Python如何創(chuàng)建裝飾器時(shí)保留函數(shù)元信息的詳細(xì)內(nèi)容,更多關(guān)于Python保留函數(shù)元信息的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. IntelliJ IDEA導(dǎo)入jar包的方法2. SSM框架JSP使用Layui實(shí)現(xiàn)layer彈出層效果3. 刪除docker里建立容器的操作方法4. IntelliJ IDEA導(dǎo)出項(xiàng)目的方法5. 解決python DataFrame 打印結(jié)果不換行問(wèn)題6. Java源碼解析之ClassLoader7. python 調(diào)用API接口 獲取和解析 Json數(shù)據(jù)8. java使用xfire搭建webservice服務(wù)的過(guò)程詳解9. Python如何測(cè)試stdout輸出10. .Net中的Http請(qǐng)求調(diào)用詳解(Post與Get)
