您好,登錄后才能下訂單哦!
本文小編為大家詳細介紹“Python裝飾器怎么正確使用”,內容詳細,步驟清晰,細節處理妥當,希望這篇“Python裝飾器怎么正確使用”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。
Python 3.6.8
要理解裝飾器之前,我們需要了解什么是閉包函數。
我們簡單寫個demo
,再解釋一下什么是閉包函數。
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()
可以看到,上述代碼所示,所謂的閉包函數是指: 閉包函數是指在函數中再定義函數,內部函數可以訪問外部的變量,在外部函數中,將內部函數作為返回值返回。
可以看到上述例子中,我們定義了exterFunc
的函數,它將接收一個形參x
,在exterFunc
函數中又中定義了innerFunc
,它也接收一個形參y
, 在innerFunc
函數中,返回x * y
,沒錯,內部函數可以訪問外部函數傳入的變量,最后將exterFunc
作為返回值返回,這就是閉包函數。
裝飾器是一種很特殊的函數,可以接收函數作為形參,且返回一個新的函數,在我們上一篇介紹生成器的時候,還記得我們使用memory_profiler
庫來打印函數的內存運行情況么? 這就是用的裝飾器。
我們可以寫個最簡單的例子,來闡述一下python
裝飾器,即:
def foo(func): def wrapper(): print("裝飾器開始運行了") func() print("裝飾器結束運行了") return wrapper @foo def sayHello(): print("hello pdudo in juejin") def main() -> None: sayHello() if __name__ == '__main__': main()
上面代碼,我們定義了一個裝飾器foo
,foo
需要傳入一個函數, foo
內部有一個函數wrapper
。這樣的函數中包函數,我們將其稱之為閉包函數,后面會介紹閉包函數。言歸正傳,在wrapper
函數中,我們可以在運行func
函數的時候,再其執行前后語句。
需要調用裝飾器的時候,只需要@
加上函數名稱即可。
要解釋這個問題,我們可以看來了解下,裝飾器解決了一些什么問題:
解決代碼重復性,對于經常需要實現類似的功能而言,可以將該功能抽離出來,作為裝飾器來調用,從而避免代碼重復。
增強代碼可讀性,在不修改原始代碼的前提下,可以利用裝飾器在函數前后增加代碼,例如 處理異常、記錄日志等等,可以利用裝飾器將附加功能和函數主要功能分開,增加代碼可讀性。
說了那么多,我們來列舉一個最簡單的例子,利用裝飾器打印一下函數的運行時間。
import time def getExecTimers(func): def wrapper(): startTimes = time.time() func() endTimes = time.time() print("函數運行時間: " , endTimes - startTimes ,"s") return wrapper @getExecTimers def testFunc(): print("開始執行函數") time.sleep(5) print("函數執行結束") def main() -> None: testFunc() if __name__ == '__main__': main()
這個裝飾器,會記錄函數的運行時間。可以看到,我們為這個函數增加了一個附屬功能,但是又沒有修改到原始函數。
上述案例,應該可以證明為什么需要使用裝飾器了。
上述我們討論了最簡單的裝飾器寫法,并且寫了一個小功能,即打印函數的運行時間。接下來,我們要看下裝飾器的其他寫法。
還記得上面我們調用裝飾器,是使用的@
+裝飾器名稱么? 其實這是python
的語法糖,如果不是用語法糖的話,應該是這樣來使用的:
def foo(func): def wrapper(): print("裝飾器開始運行了") func() print("裝飾器結束運行了") return wrapper def sayHello(): print("hello pdudo in juejin") def main() -> None: f1 = sayHello f2 = foo(f1) f2() if __name__ == '__main__': main()
完整的寫法應該如下代碼所示,這是一個完整的閉包調用邏輯。
f1 = sayHello f2 = foo(f1) f2()
而在函數前加上@
+裝飾器名稱, 是一種python
的語法糖
這里要做一個鋪墊,在python
中,有2個特殊的變量,分別為*args
和**kwargs
,都是用來處理不定量參數的,分別代表的含義為:
*args
: 將會將參數打包為元組
**kwargs
: 將會打包字典傳遞給函數
def foo(func): def wrapper(*args,**kwargs): print("裝飾器開始運行了") print("裝飾器捕獲到的參數: " ,args,**kwargs) func(*args,**kwargs) print("裝飾器結束運行了") return wrapper @foo def sayHello(a,b,c,dicts): print("傳入的參數: " , a,b,c) print("傳入的參數: " , dicts) def main() -> None: sayHello(1,2,3,{"name":"juejin"}) if __name__ == '__main__': main()
在裝飾器中,若我們要給函數傳遞參數,是需要先將參數傳遞給裝飾器,而在裝飾器中接收后再進行傳遞的,所以代碼才會是這樣的:
def foo(func): def wrapper(*args,**kwargs): print("裝飾器開始運行了") print("裝飾器捕獲到的參數: " ,args,**kwargs) func(*args,**kwargs) print("裝飾器結束運行了")
首先,我們在做傳遞調用的時候,wrapper
應該調用形參來接收,接收后,再進行傳遞給函數func
。
讀到這里,這篇“Python裝飾器怎么正確使用”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。