如何實(shí)現(xiàn)一個(gè)python函數(shù)裝飾器(Decorator)
裝飾器本質(zhì)上是一個(gè) Python 函數(shù)或類,它可以讓其他函數(shù)或類在不需要做任何代碼修改的前提下增加額外功能,裝飾器的返回值也是一個(gè)函數(shù)/類對(duì)象。它經(jīng)常用于為已有函數(shù)/類添加記錄日志、計(jì)時(shí)統(tǒng)計(jì)、性能測(cè)試等。
首先定義一個(gè)倒計(jì)時(shí)函數(shù),這個(gè)函數(shù)的功能非常簡(jiǎn)單,就是把n從當(dāng)前值減少到0。
def countdown(n): while n > 0: print('time' + str(n)) n -= 1 print(countdown.__name__)
程序輸出:
countdown
1.為函數(shù)增加一個(gè)日志裝飾器
假設(shè)現(xiàn)在要增強(qiáng)countdown的功能,在函數(shù)調(diào)用前后自動(dòng)打印日志,又不想修改函數(shù)自身的功能。這種在代碼運(yùn)行期間動(dòng)態(tài)增加功能的方式,稱之為裝飾器(Decorator)。
能打印日志的decorator,可以定義如下:
def log(func): def wrapper(*args, **kw): print('call %s().' % func.__name__) return func(*args, **kw) return wrapper
然后我們借助Python的@語(yǔ)法,把decorator置于函數(shù)的定義處:
@log def countdown(n): while n > 0: print('time:' + str(n)) n -= 1 countdown(10)
程序輸出:
call countdown().
time:10
time:9
time:8
time:7
time:6
time:5
time:4
time:3
time:2
time:1
但此時(shí)我們?cè)俅蛴『瘮?shù)的name:
print(countdown.__name__)
程序輸出:
wrapper
我們發(fā)現(xiàn)函數(shù)的元數(shù)據(jù)信息變了,這顯然不是我們想要的結(jié)果。
2. 在裝飾器中拷貝元數(shù)據(jù)
為了把函數(shù)的元數(shù)據(jù)信息都保留下來(lái),我們可以直接使用Python提供的functools庫(kù)中的@wraps裝飾器。
from functools import wraps def log(func): @wraps(func) def wrapper(*args, **kw): print('call %s().' % func.__name__) return func(*args, **kw) return wrapper @log def countdown(n): while n > 0: print('time:' + str(n)) n -= 1 print(countdown.__name__)
程序輸出:
countdown
3.為函數(shù)增加一個(gè)計(jì)時(shí)裝飾器
添加函數(shù)裝飾器的方法已經(jīng)講清楚了,現(xiàn)在再實(shí)現(xiàn)一個(gè)完整的函數(shù)計(jì)時(shí)耗時(shí)裝飾器。
import time from functools import wraps def TimeCost(func): @wraps(func) def wrapper(*arg, **kwargs): start = time.time() result = func(*args, **kwargs) end = time.time() print(func.__name__, end - start) return result return wrapper @TimeCost def countdown(n): while n > 0: print('time:' + str(n)) n -= 1 countdown(10000)
函數(shù)輸出:
('countdown', 0.0004801750183105469)
參考資料:
https://www.liaoxuefeng.com/wiki/1016959663602400/1017451662295584
以上就是如何實(shí)現(xiàn)一個(gè)python函數(shù)裝飾器(Decorator)的詳細(xì)內(nèi)容,更多關(guān)于python函數(shù)裝飾器的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Python學(xué)習(xí)筆記(一)(基礎(chǔ)入門之環(huán)境搭建)
本系列為Python學(xué)習(xí)相關(guān)筆記整理所得,IT人,多學(xué)無(wú)害,多多探索,激發(fā)學(xué)習(xí)興趣,開拓思維,不求高大上,只求懂點(diǎn)皮毛,作為知識(shí)儲(chǔ)備,不至于落后太遠(yuǎn)。本文主要介紹Python的相關(guān)背景,環(huán)境搭建。2014-06-06Empty test suite.(PyCharm程序運(yùn)行錯(cuò)誤的解決方法)
今天小編就為大家分享一篇Empty test suite.(PyCharm程序運(yùn)行錯(cuò)誤的解決方法),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-11-11?分享一個(gè)Python?遇到數(shù)據(jù)庫(kù)超好用的模塊
這篇文章主要介紹了?分享一個(gè)Python?遇到數(shù)據(jù)庫(kù)超好用的模塊,SQLALchemy這個(gè)模塊,該模塊是Python當(dāng)中最有名的ORM框架,該框架是建立在數(shù)據(jù)庫(kù)API之上,使用關(guān)系對(duì)象映射進(jìn)行數(shù)據(jù)庫(kù)的操作,,需要的朋友可以參考下2022-04-04python模型集成知識(shí)點(diǎn)總結(jié)
在本篇文章里小編給大家整理了一篇關(guān)于python模型集成知識(shí)點(diǎn)總結(jié),有需要的朋友們可以學(xué)習(xí)參考下。2021-08-08Python通過(guò)cv2讀取多個(gè)USB攝像頭
這篇文章主要為大家詳細(xì)介紹了Python通過(guò)cv2讀取多個(gè)USB攝像頭,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-08-08python GUI庫(kù)圖形界面開發(fā)之PyQt5下拉列表框控件QComboBox詳細(xì)使用方法與實(shí)例
這篇文章主要介紹了python GUI庫(kù)圖形界面開發(fā)之PyQt5下拉列表框控件QComboBox詳細(xì)使用方法與實(shí)例,需要的朋友可以參考下2020-02-02