這篇文章將為大家詳細(xì)講解有關(guān)python裝飾器如何使用,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。
創(chuàng)新互聯(lián)建站成立于2013年,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項目成都網(wǎng)站制作、網(wǎng)站建設(shè)網(wǎng)站策劃,項目實施與項目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元寧城做網(wǎng)站,已為上家服務(wù),為寧城各地企業(yè)和個人服務(wù),聯(lián)系電話:028-86922220
裝飾者模式是常用的軟件設(shè)計模式之一。通過此設(shè)計模式,我們能夠在不修改任何底層代碼情況下,給已有對象賦予新的職責(zé)。python中可以用裝飾器簡單地實現(xiàn)裝飾者模式。
1.1 將函數(shù)作為參數(shù)傳遞
在C/C++中,函數(shù)指針可以將函數(shù)作為參數(shù)傳遞給另一函數(shù)。而在python中,函數(shù)也是對象的一種,函數(shù)可以被引用,也可直接作為參數(shù)傳入函數(shù),以及作為容器對象的元素。python中可以采用如下方法實現(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 裝飾器
上面的實現(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 #等價于add = log(add) @log def add(x, y): result = x+y return result if __name__ == '__main__': print(add(1,2)) print(add.__name__)
運行情況如下:
>>print(add(1,2)) add has been called 3 >>print(add.__name__) wrapper
但上述方法亦有缺陷,原函數(shù)add的元數(shù)據(jù)(比如名字、文檔字符串、注解和參數(shù)簽名)會丟失。為避免缺陷,任何時候你定義裝飾器的時候,都應(yīng)該使用functools庫中的@wraps裝飾器來注解底層包裝函數(shù)(代碼中注釋部分)。@wraps有一個重要特征是它能讓你通過屬性 __wrapped__ 直接訪問被包裝函數(shù)。
改進(jì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)
但若使用了多個裝飾器, __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__('')
運行情況如下:
>>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ù)的。
裝飾器處理過程與下面是等價的:
normalize = log('run')(normalize)
關(guān)于python裝飾器如何使用就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
新聞標(biāo)題:python裝飾器如何使用
本文地址:http://vcdvsql.cn/article16/peidgg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供域名注冊、虛擬主機(jī)、網(wǎng)站設(shè)計、定制開發(fā)、面包屑導(dǎo)航、網(wǎng)站維護(hù)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)