Python裝飾器decorator用法實(shí)例
本文實(shí)例講述了Python裝飾器decorator用法。分享給大家供大家參考。具體分析如下:
1. 閉包(closure)
閉包是Python所支持的一種特性,它讓在非global scope定義的函數(shù)可以引用其外圍空間中的變量,這些外圍空間中被引用的變量叫做這個(gè)函數(shù)的環(huán)境變量。環(huán)境變量和這個(gè)非全局函數(shù)一起構(gòu)成了閉包。
y = [1,2,3]
def inner():
print x
print y
return inner
x = 5 #這個(gè)x沒(méi)有被引用
f = outer(2)
f()
print f.__closure__ #函數(shù)屬性__closure__存儲(chǔ)了函數(shù)的環(huán)境變量 def entrance(func):
= 5 #這個(gè)x沒(méi)有被引用f = outer(2)f()print f.__closure__ #函數(shù)屬性__closure__存儲(chǔ)了函數(shù)的環(huán)境變量 def entrance(func):
x和y都是屬于函數(shù)outer命名空間的,在inner中被引用,當(dāng)outer函數(shù)退出后,outer的命名空間不存在了,但是inner依然維護(hù)了其定義時(shí)候?qū)ζ渫獠孔兞縳,y的連接。
程序輸出:
2
[1, 2, 3]
(, )
裝飾器是一個(gè)可調(diào)用對(duì)象(a callable),在Python中,函數(shù)是對(duì)象,當(dāng)然也是可調(diào)用的,所以裝飾器可以是一個(gè)函數(shù),我們稱(chēng)其為函數(shù)裝飾器。
這個(gè)可調(diào)用對(duì)象以一個(gè)函數(shù)作為參數(shù),閉且返回另一個(gè)函數(shù)(來(lái)替換參數(shù)那個(gè)函數(shù))。
比如:
def inner():
print "inside function :", func.__name__
func()
return inner
entrance是一個(gè)裝飾器,它是一個(gè)函數(shù),它可以接收一個(gè)函數(shù)func作為參數(shù),返回了另一個(gè)函數(shù)inner。
那為什么叫裝飾器了,在返回函數(shù)inner()的內(nèi)部,調(diào)用了func(),而且還作了額外的操作,相當(dāng)于“裝飾”了函數(shù)func。
那如何使用裝飾器?
pass
fun1 = entrance(fun1)
def fun2():
pass
fun2 = entrance(fun2)
fun1,fun2的名字都沒(méi)有變,但是通過(guò)調(diào)用函數(shù)裝飾器entrance(),它們已經(jīng)指向了另一個(gè)函數(shù)inner(),“裝飾了”自己。
@操作符
Python提供的@符號(hào),實(shí)質(zhì)上就是上面做的,對(duì)一個(gè)函數(shù)名進(jìn)行從新賦值,是語(yǔ)法上的技巧。所以上面的代碼等價(jià)于
def fun1():
pass
@entrance
def fun2():
pass
2. 裝飾器的用途
從這個(gè)刻意構(gòu)造的很簡(jiǎn)單的例子,可以看出裝飾器的意義,如果一個(gè)函數(shù)需要一個(gè)功能,如果這個(gè)功能可以被使用在很多函數(shù)上,或是函數(shù)并不是自己實(shí)現(xiàn),那可以寫(xiě)個(gè)裝飾器來(lái)實(shí)現(xiàn)這些功能。
上面的裝飾器entrance,裝飾一個(gè)函數(shù)后,函數(shù)被調(diào)用時(shí)會(huì)打印出這個(gè)函數(shù)的名字。
但是有一個(gè)問(wèn)題,這個(gè)裝飾器從功能上看,是要應(yīng)該可以用來(lái)裝飾任何函數(shù),但是如果我們用它來(lái)裝飾了一個(gè)帶參數(shù)的函數(shù)
def fun3(x):
pass
只要不調(diào)用fun3,這三行代碼是不會(huì)讓Python解釋器報(bào)錯(cuò)的,因?yàn)槲覀円呀?jīng)知道,它等價(jià)于:
pass
fun3 = entrance(fun3)
我們定義了一個(gè)帶參的函數(shù)fun3,然后把fun3指向了另一個(gè)函數(shù)inner(),當(dāng)然不會(huì)有什么錯(cuò)。
但是,當(dāng)我們使用fun3時(shí),我們肯定會(huì)按照它定義時(shí)的樣子去使用它,給它傳入一個(gè)參數(shù)。
>>>fun3(1)
這里就會(huì)出錯(cuò)了,看看解釋器怎么報(bào)錯(cuò)的
Traceback (most recent call last):
File “decorator.py”, line 23, in chabaoo.cn <module>
fun3(1)
TypeError: inner() takes no arguments (1 given)
當(dāng)然我們已經(jīng)很容易知到為什么會(huì)這樣報(bào)錯(cuò)了,fun3已經(jīng)不是指向它定義時(shí)那個(gè)函數(shù)了,它現(xiàn)在指向了”inner()”,而inner是沒(méi)有參數(shù)的,當(dāng)然會(huì)出錯(cuò)。
那怎么解決呢?
修改一下inner()的定義,讓它可以就收任意個(gè)參數(shù)就可以了。
def inner(*args, **kvargs):
print "inside function : ", func.__name__
func(*args, **kvargs)
return inner
現(xiàn)在,給inner傳任意個(gè)參數(shù)都不會(huì)出錯(cuò)了,也就是entrance可以被用來(lái)裝飾任何一個(gè)函數(shù)了。
3. 寫(xiě)個(gè)裝飾器logger
一個(gè)函數(shù)被調(diào)用時(shí),在日志里記錄其名稱(chēng)和被調(diào)用的實(shí)際參數(shù)
def inner(*args, **kvargs):
print func.__name__, 'called, arguments: ', args, kvargs
func(*args, **kvargs)
return inner
希望本文所述對(duì)大家的Python程序設(shè)計(jì)有所幫助。
相關(guān)文章
利用pygame完成動(dòng)畫(huà)精靈和碰撞檢測(cè)
這篇文章主要介紹了利用pygame完成動(dòng)畫(huà)精靈和碰撞檢測(cè),代碼詳細(xì),內(nèi)容豐富,對(duì)于想要學(xué)習(xí)pygame的朋友來(lái)講是一個(gè)不錯(cuò)的練習(xí),需要的朋友可以參考下2021-04-04Python利用全連接神經(jīng)網(wǎng)絡(luò)求解MNIST問(wèn)題詳解
這篇文章主要介紹了Python利用全連接神經(jīng)網(wǎng)絡(luò)求解MNIST問(wèn)題,結(jié)合實(shí)例形式詳細(xì)分析了單隱藏層神經(jīng)網(wǎng)絡(luò)與多層神經(jīng)網(wǎng)絡(luò),以及Python全連接神經(jīng)網(wǎng)絡(luò)求解MNIST問(wèn)題相關(guān)操作技巧,需要的朋友可以參考下2020-01-01python如何實(shí)現(xiàn)數(shù)組元素兩兩相加
這篇文章主要介紹了python如何實(shí)現(xiàn)數(shù)組元素兩兩相加,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-05-05python編程PyQt5創(chuàng)建按鈕及觸發(fā)點(diǎn)擊事件示例解析
這篇文章主要為大家介紹了python編程使用PyQt5如何創(chuàng)建按鈕及觸發(fā)點(diǎn)擊事件的示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-10-10Python?Setuptools的?setup.py實(shí)例詳解
setup.py是一個(gè)?python?文件,它的存在表明您要安裝的模塊/包可能已經(jīng)用?Setuptools?打包和分發(fā),這是分發(fā)?Python?模塊的標(biāo)準(zhǔn)。?它的目的是正確安裝軟件,本文給大家講解Python?Setuptools的?setup.py感興趣的朋友跟隨小編一起看看吧2022-12-12pygame.display.flip()和pygame.display.update()的區(qū)別及說(shuō)明
這篇文章主要介紹了pygame.display.flip()和pygame.display.update()的區(qū)別及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03簡(jiǎn)單分析Python中用fork()函數(shù)生成的子進(jìn)程
這篇文章主要介紹了Python中用fork()函數(shù)生成的子進(jìn)程,分析子進(jìn)程與父進(jìn)程的執(zhí)行順序,需要的朋友可以參考下2015-05-05基于Python新建用戶并產(chǎn)生隨機(jī)密碼過(guò)程解析
這篇文章主要介紹了基于Python新建用戶并產(chǎn)生隨機(jī)密碼過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-10-10