您好,登錄后才能下訂單哦!
這篇文章將為大家詳細(xì)講解有關(guān)python裝飾器如何使用,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。
裝飾者模式是常用的軟件設(shè)計(jì)模式之一。通過此設(shè)計(jì)模式,我們能夠在不修改任何底層代碼情況下,給已有對象賦予新的職責(zé)。python中可以用裝飾器簡單地實(shí)現(xiàn)裝飾者模式。
1.1 將函數(shù)作為參數(shù)傳遞
在C/C++中,函數(shù)指針可以將函數(shù)作為參數(shù)傳遞給另一函數(shù)。而在python中,函數(shù)也是對象的一種,函數(shù)可以被引用,也可直接作為參數(shù)傳入函數(shù),以及作為容器對象的元素。python中可以采用如下方法實(shí)現(xiàn)裝飾者模式:
#!/usr/bin/env python3.6 # -*- coding: utf-8 -*- def add(x, y): result = x+y return result def log(func): def wrapper(*args, **kwargs): result = func(*args) print(func.__name__,'has been called\n') return result return wrapper if __name__ == '__main__': print(log(add)(1,2))
上述代碼中,log函數(shù)以需要被裝飾的函數(shù)作為參數(shù),并返回函數(shù)對象。被返回的函數(shù)的參數(shù)為可變參數(shù)*args與**kwargs(*args參數(shù)會被封裝成tuple,**kwargs參數(shù)則會被封裝成字典對象),以適應(yīng)不同函數(shù)的不同參數(shù),保證通用性。
1.2 裝飾器
上面的實(shí)現(xiàn)方法有些繁雜,所有調(diào)用被裝飾的函數(shù)之處的代碼,都要進(jìn)行相應(yīng)修改,自然不符合python簡潔易讀的特性。因此python中給出相應(yīng)語法糖來增加可讀性和易用性,那便是“裝飾器”。
from functools import wraps def log(func): #@wraps(func) def wrapper(*args, **kwargs): result = func(*args) print(func.__name__,'has been called') return result return wrapper #等價(jià)于add = log(add) @log def add(x, y): result = x+y return result if __name__ == '__main__': print(add(1,2)) print(add.__name__)
運(yùn)行情況如下:
>>print(add(1,2)) add has been called 3 >>print(add.__name__) wrapper
但上述方法亦有缺陷,原函數(shù)add的元數(shù)據(jù)(比如名字、文檔字符串、注解和參數(shù)簽名)會丟失。為避免缺陷,任何時(shí)候你定義裝飾器的時(shí)候,都應(yīng)該使用functools庫中的@wraps裝飾器來注解底層包裝函數(shù)(代碼中注釋部分)。@wraps有一個(gè)重要特征是它能讓你通過屬性 __wrapped__ 直接訪問被包裝函數(shù)。
改進(jìn)后運(yùn)行情況:
>>print(add(1,2)) add has been called 3 >>print(add.__name__) add
1.3 解除裝飾器
當(dāng)裝飾器已經(jīng)作用于某函數(shù),而你想撤銷它,那么可以訪問 __wrapped__屬性來訪問原始函數(shù)
orig_add = add.__wrapped__ orig_add(1,2)
但若使用了多個(gè)裝飾器, __wrapped__屬性會變得不可控,應(yīng)盡量避免使用。
若有如下代碼:
#!/usr/bin/env python3.6 # -*- coding: utf-8 -*- import functools import time def metric(func): @functools.wraps(func) def wrapper(*args,**kv): print('Decorator1') f = func(*args,**kv) return f return wrapper def logging(func): @functools.wraps(func) def wrapper(*args,**kv): print('Decorator2') f = func(*args,**kv) return f return wrapper @metric @logging def normalize(name): sName = name[0:1].upper() + name[1:].lower() print(sName) if __name__ == '__main__': normalize('heLlO') normalize.__wrapper__('')
運(yùn)行情況如下:
>>normalize('helLo') Decorator1 Decorator2 Hello >>normalize.__wrapped__('world') Decorator2 World
1.4 定義帶參數(shù)的裝飾器
from functools import wraps def log(text): def decorator(func): @wraps(func) def wrappering(*args,**kv): print('%s %s():'%(text,func.__name__)) return func(*args,**kv) return wrappering return decorator @log('run') def normalize(name): sName = name[0:1].upper() + name[1:].lower() print(sName)
裝飾器函數(shù)可以帶參數(shù),最外層的函數(shù)會將參數(shù)傳給內(nèi)層的裝飾器函數(shù),即wrappering函數(shù)是可以使用log的傳入?yún)?shù)的。
裝飾器處理過程與下面是等價(jià)的:
normalize = log('run')(normalize)
關(guān)于python裝飾器如何使用就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。