Python 中的裝飾器

2021-08-08 15:27:07 字數 3224 閱讀 1665

1.

基本概念

問題1:裝飾器是什麼?

解答: 嚴格來說,裝飾器只是語法糖, 裝飾器是可呼叫的物件,可以像常規的可呼叫物件那樣呼叫,特殊的地方是裝飾器的引數是乙個函式

問題2:裝飾器有什麼特性?

解答: 裝飾器有2個特性,一是可以把被裝飾的函式替換成其他函式, 二是可以在載入模組時候立即執行

def decorate(func):

print('running decorate', func)

def decorate_inner():

print('running decorate_inner function')

return func()

return decorate_inner

@decorate

def func_1():

print('running func_1')

if __name__ == '__main__':

print(func_1)

#返回值

running decorate .decorate_inner at 0x7f29f641cb70>

問題3:如何使用被裝飾函式中的引數?

解答: 通過args 和 *kwargs 傳遞被修飾函式中的引數

def decorate(func):

def decorate_inner(*args, **kwargs):

print(type(args), type(kwargs))

print('args', args, 'kwargs', kwargs)

return func(*args, **kwargs)

return decorate_inner

@decorate

def func_1(*args, **kwargs):

print(args, kwargs)

if __name__ == '__main__':

func_1('1', '2', '3', para_1='1', para_2='2', para_3='3')

#返回值

args ('1', '2', '3') kwargs

('1', '2', '3')

2.疊放裝飾器

問題1:疊放裝飾器執行順序是什麼?

解答: 如果乙個函式被多個裝飾器修飾,其實應該是該函式先被最裡面的裝飾器修飾後(下面例子中函式main()先被inner裝飾,變成新的函式),變成另乙個函式後,再次被裝飾器修飾

def outer(func):

print('enter outer', func)

print('running outer')

func()

def inner(func):

print('enter inner', func)

print('running inner')

func()

@outer

@inner

def main():

print('running main')

if __name__ == '__main__':

main()

#返回值

running outer

running inner

running main

3.標準庫中的裝飾器

問題1: 標準庫中都有哪些裝飾器?

解答: 標準庫中有多種裝飾器, 例如:裝飾方法的函式有property, classmethod, staticmethod; functools模組中的lru_cache, singledispatch,  wraps 等等

from functools import lru_cache

from functools import singledispatch

from functools import wraps

問題2:為什麼要使用@wraps裝飾器?它的作用是什麼?

解答: 使用裝飾器會產生我們可能不希望出現的***, 例如:改變被修飾函式名稱,對於偵錯程式或者物件序列化器等需要使用內省機制的那些工具,可能會無法正常執行;其實呼叫裝飾器後,會將同乙個作用域中原來函式同名的那個變數(例如下面的func_1),重新賦值為裝飾器返回的物件;使用@wraps後,會把與內部函式(被修飾函式,例如下面的func_1)相關的重要元資料全部複製到外圍函式(例如下面的decorate_inner)

from functools import wraps

def decorate(func):

print('running decorate', func)

@wraps(func)

def decorate_inner():

print('running decorate_inner function', decorate_inner)

return func()

return decorate_inner

@decorate

def func_1():

print('running func_1', func_1)

if __name__ == '__main__':

func_1()

#返回值

running decorate running decorate_inner function running func_1

4. 裝飾器設計模式

問題1: 什麼是裝飾器設計模式?

解答: 動態的給乙個物件新增一些額外的職責,就擴充套件功能而言,裝飾器模式比子類化更加靈活,在設計模式中,裝飾器和元件都是抽象類,為了給具體的元件新增行為,具體的裝飾器例項要包裝具體元件的例項,即,裝飾器和所裝飾的元件介面一致,對使用該組建的客戶透明,將客戶的請求**給該元件,並且可能在**前後執行一些額外的操作,透明性使得可以遞迴巢狀多個裝飾器,從而可以新增任意多個功能

問題2: python中的裝飾器函式和設計模式中的裝飾器模式有什麼關係?

解答:  修飾器模式和python修飾器之間並不是一對一的等價關係, python裝飾器函式更為強大,不僅僅可以實現裝飾器模式。

python中的裝飾器

其實去年就開始學python了,零零散散,陸陸續續學了點,期間學習了python web開發,爬蟲系統 但是一些基礎性的知識點長時間不看了就會忘,所以寫個部落格記錄下來,忘了可以隨時檢視,不用回去看 了,希望也能幫助大家學習哈。python中的裝飾器decorator其實是乙個高階函式,它接受乙個函...

python中的 裝飾器

示例 def fun a arg print a arg fun a deffun b print b 輸出 c python34 python.exe c users administrator desktop test.pyab process finished with exit code 0...

python中的裝飾器

什麼是裝飾器 如果有函式a,b,c,已經所有編寫完畢。這時你發現a,b,c都須要同乙個功能,這時該怎麼辦?答 裝飾器 裝飾器事實上就是乙個函式,只是這個函式的返回值是乙個函式 個人理解。裝飾器主要就是為了完畢上邊的這個功能,將a,b,c 函式包裹在還有乙個函式d中。d函式在a函式執行之前或之後,處理...