Python 在函數(shù)上添加包裝器
問(wèn)題
你想在函數(shù)上添加一個(gè)包裝器,增加額外的操作處理(比如日志、計(jì)時(shí)等)。
解決方案
如果你想使用額外的代碼包裝一個(gè)函數(shù),可以定義一個(gè)裝飾器函數(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
下面是使用裝飾器的例子:
>>> @timethis... def countdown(n):... ’’’... Counts down... ’’’... while n > 0:... n -= 1...>>> countdown(100000)countdown 0.008917808532714844>>> countdown(10000000)countdown 0.87188299392912>>>
討論
一個(gè)裝飾器就是一個(gè)函數(shù),它接受一個(gè)函數(shù)作為參數(shù)并返回一個(gè)新的函數(shù)。當(dāng)你像下面這樣寫:
@timethisdef countdown(n): pass
跟像下面這樣寫其實(shí)效果是一樣的:
def countdown(n): passcountdown = timethis(countdown)
順便說(shuō)一下,內(nèi)置的裝飾器比如 @staticmethod, @classmethod,@property 原理也是一樣的。例如,下面這兩個(gè)代碼片段是等價(jià)的:
class A: @classmethod def method(cls): passclass B: # Equivalent definition of a class method def method(cls): pass method = classmethod(method)
在上面的 wrapper() 函數(shù)中,裝飾器內(nèi)部定義了一個(gè)使用 *args 和 **kwargs 來(lái)接受任意參數(shù)的函數(shù)。在這個(gè)函數(shù)里面調(diào)用了原始函數(shù)并將其結(jié)果返回,不過(guò)你還可以添加其他額外的代碼(比如計(jì)時(shí))。然后這個(gè)新的函數(shù)包裝器被作為結(jié)果返回來(lái)代替原始函數(shù)。
需要強(qiáng)調(diào)的是裝飾器并不會(huì)修改原始函數(shù)的參數(shù)簽名以及返回值。使用 *args 和 **kwargs 目的就是確保任何參數(shù)都能適用。而返回結(jié)果值基本都是調(diào)用原始函數(shù) func(*args, **kwargs) 的返回結(jié)果,其中func就是原始函數(shù)。
剛開(kāi)始學(xué)習(xí)裝飾器的時(shí)候,會(huì)使用一些簡(jiǎn)單的例子來(lái)說(shuō)明,比如上面演示的這個(gè)。不過(guò)實(shí)際場(chǎng)景使用時(shí),還是有一些細(xì)節(jié)問(wèn)題要注意的。比如上面使用 @wraps(func) 注解是很重要的,它能保留原始函數(shù)的元數(shù)據(jù)(下一小節(jié)會(huì)講到),新手經(jīng)常會(huì)忽略這個(gè)細(xì)節(jié)。接下來(lái)的幾個(gè)小節(jié)我們會(huì)更加深入的講解裝飾器函數(shù)的細(xì)節(jié)問(wèn)題,如果你想構(gòu)造你自己的裝飾器函數(shù),需要認(rèn)真看一下。
以上就是Python 在函數(shù)上添加包裝器的詳細(xì)內(nèi)容,更多關(guān)于Python 添加包裝器的資料請(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. java使用xfire搭建webservice服務(wù)的過(guò)程詳解6. .Net中的Http請(qǐng)求調(diào)用詳解(Post與Get)7. JS如何在數(shù)組指定位置插入元素8. PHP下對(duì)緩沖區(qū)的控制9. Vue 實(shí)現(xiàn)對(duì)quill-editor組件中的工具欄添加title10. Java源碼解析之ClassLoader
