Python裝飾器的定義和使用詳情
1.裝飾器的定義
裝飾器:給已有函數(shù)增加額外的功能的函數(shù),本質(zhì)上是一個(gè)閉包函數(shù)
特點(diǎn):
- 1.不修改已有函數(shù)的源代碼
- 2.不修改已有函數(shù)的調(diào)用方式
- 3.給已有函數(shù)增加額外的功能
- 4.代碼執(zhí)行時(shí)先解析裝飾器
import time ? # 裝飾器原理 # def show(): # ? ? n=0 # ? ? for i in range(10000000): # ? ? ? ? n+=i # ? ? print('show_',n) # # # 定義一個(gè)閉包 # def count_time(fun): # ? ? def inner(): # ? ? ? ? start=time.time() # ? ? ? ? fun() # ? ? ? ? end=time.time() # ? ? ? ? print(f'用時(shí){end-start}秒') # ? ? return inner # # # 裝飾器在裝飾函數(shù)時(shí)的原理 # show=count_time(show) # show() ? # 定義裝飾器(語(yǔ)法糖) def count_time(fun): ? ?# 必須要有一個(gè)參數(shù)接收被裝飾函數(shù) ? ? def inner(): ? ? ? ? start=time.time() ? ? ? ? fun() ? ? ? ? end=time.time() ? ? ? ? print(f'用時(shí){end-start}秒') ? ? return inner # 裝飾器寫(xiě)法:@閉包的外部函數(shù),必須在閉包定以后使用 print('解析裝飾器1') @count_time # 解釋成show=count_time(show),show指向count_time函數(shù)中的inner def show(): ? ? n=0 ? ? for i in range(10000000): ? ? ? ? n+=i ? ? print('show_',n) ? print('解析裝飾器2') @count_time # 解釋成display=count_time(display) def display(): ? ? print('Display') ? print('正式執(zhí)行...') show() display()
2.裝飾器的通用類型的定義
(當(dāng)被裝飾函數(shù)有參數(shù)或者有返回值時(shí)同樣適用)
''' 裝飾器的通用類型的定義(當(dāng)被裝飾函數(shù)有參數(shù)或者有返回值時(shí)同樣適用) ''' ? def outer(func): ? ? def inner(*args,**kwargs): ?# *為元組和列表解包,**為字典解包 ? ? ? ? print('*'*30) ? ? ? ? print(args,kwargs) ? ? ? ? ret=func(*args,**kwargs) ? ?# 解包,否則形參是元組或字典 ? ? ? ? print('*'*30) ? ? ? ? return ret ? ? return inner ? @outer def show(name,msg): ? ? return str(name)+' say: '+str(msg) ? print(show('Tom',msg='Hello'))
3.多個(gè)裝飾器同時(shí)裝飾一個(gè)函數(shù)
# 第一個(gè)閉包 def wrapper_div(func): ? ? def inner(*args,**kwargs): ? ? ? ? return '<div>'+func(*args,**kwargs)+'</div>' ? ? return inner ? # 第二個(gè)閉包 def wrapper_p(func): ? ? def inner(*args,**kwargs): ? ? ? ? return '<p>'+func(*args,**kwargs)+'</p>' ? ? return inner ? # 從下往上裝飾,從上往下執(zhí)行 @wrapper_div @wrapper_p # 定義一個(gè)函數(shù) def show(): ? ? return 'Short life I use Python.' ? print(show()) ? #<div><p>Short life I use Python.</p></div>
4.多個(gè)裝飾器同時(shí)裝飾一個(gè)函數(shù)(二)
def outer1(func): ? ? def inner(): ? ? ? ? print('裝飾器1-1') ? ? ? ? func() ? ? ? ? print('裝飾器1-2') ? ? return inner ? def outer2(func): ? ? def inner(): ? ? ? ? print('裝飾器2-1') ? ? ? ? func() ? ? ? ? print('裝飾器2-2') ? ? return inner ''' 1.show指向outer1.inner 2.outer1.inner.func指向outer2.inner 3.outer2.inner.func指向show ''' @outer1 @outer2 def show(): ? ? print('Show...') ? show()
5.類裝飾器使用方法
import time ? class Wrapper(): ? ? def __init__(self,func): ? ? ? ? self.func=func ? ? ? # 當(dāng)類中實(shí)現(xiàn)了此方法時(shí),該類的實(shí)例對(duì)象就變成了可調(diào)用對(duì)象,即可以在實(shí)例對(duì)象后面加() ? ? def __call__(self, *args, **kwargs): ? ? ? ? print('裝飾內(nèi)容1...') ? ? ? ? start=time.time() ? ? ? ? ret=self.func(*args,**kwargs) ? ? ? ? end=time.time() ? ? ? ? print(f'執(zhí)行了{(lán)end-start}秒') ? ? ? ? print('裝飾內(nèi)容2...') ? ? ? ? return ret
該裝飾器執(zhí)行完成后,被裝飾函數(shù)指向該類的實(shí)例對(duì)象
如果讓被裝飾函數(shù)執(zhí)行,那么在類中要添加__call__方法,相當(dāng)于閉包格式中的內(nèi)函數(shù)
一旦被裝飾函數(shù)執(zhí)行調(diào)用,那么就會(huì)去執(zhí)行實(shí)例對(duì)象中的__call__函數(shù)
@Wrapper ? ?#解釋成show=Wrapper(show),show變成了類的一個(gè)對(duì)象 def show(): ? ? print('Show...') ? show() 6.裝飾器帶有參數(shù)(使用帶有參數(shù)的裝飾器,其實(shí)是在裝飾器外面又包裹了一個(gè)函數(shù)) # @Author ?: Kant # @Time ? ?: 2022/1/23 22:43 ? def set_args(msg): ? ? def outer(func): ? ? ? ? def inner(): ? ? ? ? ? ? print('裝飾內(nèi)容',msg) ? ? ? ? ? ? func() ? ? ? ? return inner ? ? return outer ? ''' 使用帶有參數(shù)的裝飾器,其實(shí)是在裝飾器外面又包裹了一個(gè)函數(shù),使用該函數(shù)接收參數(shù),返回的是裝飾器 調(diào)用set_args()后會(huì)返回outer的地址引用,變成了@outer ''' @set_args('Hello') # 無(wú)論閉包函數(shù)寫(xiě)成什么樣子,被裝飾函數(shù)永遠(yuǎn)指向閉包函數(shù)的內(nèi)函數(shù) def show(): ? ? print('Show...') ? show()
6.使用裝飾器實(shí)現(xiàn)自動(dòng)維護(hù)路由表
路由功能:通過(guò)請(qǐng)求的路徑,可以找到資源的地址
# 定義一個(gè)路由表字典 router_table={} def router(url): ? ? def wrapper(func): ? ? ? ? def inner(): ? ? ? ? ? ? print('1') ? ? ? ? ? ? print('inner-',func) ? ?# 查看當(dāng)前的被裝飾函數(shù)是誰(shuí) ? ? ? ? ? ? func() ? ? ? ? # 在這里維護(hù)路由表字典 ? ? ? ? router_table[url]=inner # 如果寫(xiě)func,inner函數(shù)中的內(nèi)容都不會(huì)執(zhí)行 ? ? ? ? print('路由表字典:',router_table) ? ? ? ? return inner ? ? return wrapper ? @router('index.html') def index(): ? ? print('首頁(yè)內(nèi)容') ? @router('center.html') def center(): ? ? print('個(gè)人中心') ? @router('mail.html') def mail(): ? ? print('郵箱頁(yè)面') ? @router('login.html') def login(): ? ? print('登錄頁(yè)面') ? def error(): ? ? print('訪問(wèn)頁(yè)面不存在') ? def request_url(url): ? ? func=error ? ? if url in router_table: ? ? ? ? func=router_table[url] ? ? func() ? print('開(kāi)始執(zhí)行函數(shù)') request_url('index.html') request_url('center.html') request_url('mail.html') request_url('test.html') request_url('login.html')
到此這篇關(guān)于Python裝飾器的定義和使用詳情的文章就介紹到這了,更多相關(guān)Python裝飾器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python?實(shí)現(xiàn)syslog?服務(wù)器的詳細(xì)過(guò)程
這篇文章主要介紹了python?實(shí)現(xiàn)syslog服務(wù)器的詳細(xì)過(guò)程,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08使用Python 自動(dòng)生成 Word 文檔的教程
今天小編就為大家分享一篇使用Python 自動(dòng)生成 Word 文檔的教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-02-02caffe的python接口caffemodel參數(shù)及特征抽取示例
這篇文章主要介紹了caffe的python接口caffemodel參數(shù)及特征抽取示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06簡(jiǎn)介Python設(shè)計(jì)模式中的代理模式與模板方法模式編程
這篇文章主要介紹了Python設(shè)計(jì)模式中的代理模式與模板方法模式編程,文中舉了兩個(gè)簡(jiǎn)單的代碼片段來(lái)說(shuō)明,需要的朋友可以參考下2016-02-02Datawhale練習(xí)之二手車價(jià)格預(yù)測(cè)
此篇文章是關(guān)于Datawhale練習(xí),代碼完整,但由于該數(shù)據(jù)集中數(shù)據(jù)特征較少(39維),以下可作為少量特征情況下的分析。當(dāng)特征數(shù)目過(guò)大(成千上萬(wàn))時(shí),需要繼續(xù)學(xué)習(xí)。需要的朋友可以參考下2021-04-04Python中的Numpy?面向數(shù)組編程常見(jiàn)操作
這篇文章主要介紹了Python中的Numpy?面向數(shù)組編程常見(jiàn)操作,使用Numpy數(shù)組可以使你利用簡(jiǎn)單的數(shù)組表達(dá)式完成多項(xiàng)數(shù)據(jù)操作任務(wù),而不需要編寫(xiě)大量的循環(huán),這個(gè)極大的幫助了我們高效的解決問(wèn)題2022-07-07