您好,登錄后才能下訂單哦!
本篇內(nèi)容介紹了“Python中的裝飾器是什么及怎么使用”的有關(guān)知識(shí),在實(shí)際案例的操作過程中,不少人都會(huì)遇到這樣的困境,接下來(lái)就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!
Python 3.6.8
要理解裝飾器之前,我們需要了解什么是閉包函數(shù)。
我們簡(jiǎn)單寫個(gè)demo
,再解釋一下什么是閉包函數(shù)。
def exterFunc(x): def innerFunc(y): return x * y return innerFunc def main() -> None: f = exterFunc(6) result = f(5) print(result) if __name__ == '__main__': main()
可以看到,上述代碼所示,所謂的閉包函數(shù)是指: 閉包函數(shù)是指在函數(shù)中再定義函數(shù),內(nèi)部函數(shù)可以訪問外部的變量,在外部函數(shù)中,將內(nèi)部函數(shù)作為返回值返回。
可以看到上述例子中,我們定義了exterFunc
的函數(shù),它將接收一個(gè)形參x
,在exterFunc
函數(shù)中又中定義了innerFunc
,它也接收一個(gè)形參y
, 在innerFunc
函數(shù)中,返回x * y
,沒錯(cuò),內(nèi)部函數(shù)可以訪問外部函數(shù)傳入的變量,最后將exterFunc
作為返回值返回,這就是閉包函數(shù)。
裝飾器是一種很特殊的函數(shù),可以接收函數(shù)作為形參,且返回一個(gè)新的函數(shù),在我們上一篇介紹生成器的時(shí)候,還記得我們使用memory_profiler
庫(kù)來(lái)打印函數(shù)的內(nèi)存運(yùn)行情況么? 這就是用的裝飾器。
我們可以寫個(gè)最簡(jiǎn)單的例子,來(lái)闡述一下python
裝飾器,即:
def foo(func): def wrapper(): print("裝飾器開始運(yùn)行了") func() print("裝飾器結(jié)束運(yùn)行了") return wrapper @foo def sayHello(): print("hello pdudo in juejin") def main() -> None: sayHello() if __name__ == '__main__': main()
上面代碼,我們定義了一個(gè)裝飾器foo
,foo
需要傳入一個(gè)函數(shù), foo
內(nèi)部有一個(gè)函數(shù)wrapper
。這樣的函數(shù)中包函數(shù),我們將其稱之為閉包函數(shù),后面會(huì)介紹閉包函數(shù)。言歸正傳,在wrapper
函數(shù)中,我們可以在運(yùn)行func
函數(shù)的時(shí)候,再其執(zhí)行前后語(yǔ)句。
需要調(diào)用裝飾器的時(shí)候,只需要@
加上函數(shù)名稱即可。
要解釋這個(gè)問題,我們可以看來(lái)了解下,裝飾器解決了一些什么問題:
解決代碼重復(fù)性,對(duì)于經(jīng)常需要實(shí)現(xiàn)類似的功能而言,可以將該功能抽離出來(lái),作為裝飾器來(lái)調(diào)用,從而避免代碼重復(fù)。
增強(qiáng)代碼可讀性,在不修改原始代碼的前提下,可以利用裝飾器在函數(shù)前后增加代碼,例如 處理異常、記錄日志等等,可以利用裝飾器將附加功能和函數(shù)主要功能分開,增加代碼可讀性。
說了那么多,我們來(lái)列舉一個(gè)最簡(jiǎn)單的例子,利用裝飾器打印一下函數(shù)的運(yùn)行時(shí)間。
import time def getExecTimers(func): def wrapper(): startTimes = time.time() func() endTimes = time.time() print("函數(shù)運(yùn)行時(shí)間: " , endTimes - startTimes ,"s") return wrapper @getExecTimers def testFunc(): print("開始執(zhí)行函數(shù)") time.sleep(5) print("函數(shù)執(zhí)行結(jié)束") def main() -> None: testFunc() if __name__ == '__main__': main()
這個(gè)裝飾器,會(huì)記錄函數(shù)的運(yùn)行時(shí)間。可以看到,我們?yōu)檫@個(gè)函數(shù)增加了一個(gè)附屬功能,但是又沒有修改到原始函數(shù)。
上述案例,應(yīng)該可以證明為什么需要使用裝飾器了。
上述我們討論了最簡(jiǎn)單的裝飾器寫法,并且寫了一個(gè)小功能,即打印函數(shù)的運(yùn)行時(shí)間。接下來(lái),我們要看下裝飾器的其他寫法。
還記得上面我們調(diào)用裝飾器,是使用的@
+裝飾器名稱么? 其實(shí)這是python
的語(yǔ)法糖,如果不是用語(yǔ)法糖的話,應(yīng)該是這樣來(lái)使用的:
def foo(func): def wrapper(): print("裝飾器開始運(yùn)行了") func() print("裝飾器結(jié)束運(yùn)行了") return wrapper def sayHello(): print("hello pdudo in juejin") def main() -> None: f1 = sayHello f2 = foo(f1) f2() if __name__ == '__main__': main()
完整的寫法應(yīng)該如下代碼所示,這是一個(gè)完整的閉包調(diào)用邏輯。
f1 = sayHello f2 = foo(f1) f2()
而在函數(shù)前加上@
+裝飾器名稱, 是一種python
的語(yǔ)法糖
這里要做一個(gè)鋪墊,在python
中,有2個(gè)特殊的變量,分別為*args
和**kwargs
,都是用來(lái)處理不定量參數(shù)的,分別代表的含義為:
*args
: 將會(huì)將參數(shù)打包為元組
**kwargs
: 將會(huì)打包字典傳遞給函數(shù)
def foo(func): def wrapper(*args,**kwargs): print("裝飾器開始運(yùn)行了") print("裝飾器捕獲到的參數(shù): " ,args,**kwargs) func(*args,**kwargs) print("裝飾器結(jié)束運(yùn)行了") return wrapper @foo def sayHello(a,b,c,dicts): print("傳入的參數(shù): " , a,b,c) print("傳入的參數(shù): " , dicts) def main() -> None: sayHello(1,2,3,{"name":"juejin"}) if __name__ == '__main__': main()
在裝飾器中,若我們要給函數(shù)傳遞參數(shù),是需要先將參數(shù)傳遞給裝飾器,而在裝飾器中接收后再進(jìn)行傳遞的,所以代碼才會(huì)是這樣的:
def foo(func): def wrapper(*args,**kwargs): print("裝飾器開始運(yùn)行了") print("裝飾器捕獲到的參數(shù): " ,args,**kwargs) func(*args,**kwargs) print("裝飾器結(jié)束運(yùn)行了")
首先,我們?cè)谧鰝鬟f調(diào)用的時(shí)候,wrapper
應(yīng)該調(diào)用形參來(lái)接收,接收后,再進(jìn)行傳遞給函數(shù)func
。
“Python中的裝飾器是什么及怎么使用”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。