Python中的裝飾器是一種高級(jí)編程技巧,通過它我們可以在不改變?cè)泻瘮?shù)的基礎(chǔ)上,在其前后添加一些額外的邏輯。這使得我們可以很方便地?cái)U(kuò)展原有代碼的功能,更好地實(shí)現(xiàn)代碼重用和維護(hù)。
一個(gè)裝飾器本質(zhì)上就是一個(gè)函數(shù),它接受一個(gè)函數(shù)作為參數(shù),并返回一個(gè)新函數(shù)。新函數(shù)實(shí)現(xiàn)了對(duì)原有函數(shù)的裝飾(即添加了額外的邏輯)。
@decorator
def func():
# do something
在裝飾器的語法糖@decorator后緊跟著要裝飾的函數(shù)func,在函數(shù)定義之前。這樣,當(dāng)我們?cè)谡{(diào)用func()時(shí),實(shí)際上會(huì)調(diào)用裝飾后的新函數(shù)。
在使用裝飾器時(shí),需要注意一些細(xì)節(jié):
def my_decorator(func):
def wrapper():
# do something before func() is called
func()
# do something after func() is called
return wrapper
@my_decorator
def say_hello():
print("Hello World!")
say_hello()
裝飾器函數(shù)的參數(shù)是被裝飾的函數(shù),返回值是一個(gè)新函數(shù)。在新函數(shù)中,可以在被裝飾的函數(shù)執(zhí)行前后添加額外的邏輯。在上述示例中,my_decorator是一個(gè)裝飾器函數(shù),它的參數(shù)func是一個(gè)函數(shù),返回值wrapper也是一個(gè)函數(shù)。在wrapper函數(shù)中,我們?cè)谠械膕ay_hello函數(shù)前后添加了額外的邏輯,這樣當(dāng)我們調(diào)用say_hello函數(shù)時(shí),實(shí)際上會(huì)調(diào)用裝飾后的新函數(shù)wrapper。
除了使用函數(shù)實(shí)現(xiàn)裝飾器外,我們還可以使用類來實(shí)現(xiàn)裝飾器。
class my_decorator(object):
def __init__(self, func):
self.func = func
def __call__(self):
# do something before self.func() is called
self.func()
# do something after self.func() is called
@my_decorator
def say_hello():
print("Hello World!")
say_hello()
在使用類實(shí)現(xiàn)裝飾器時(shí),我們需要實(shí)現(xiàn)__init__和__call__方法。__init__方法初始化裝飾器,__call__方法實(shí)現(xiàn)裝飾器的邏輯。在上述示例中,my_decorator是一個(gè)裝飾器類,它的__init__方法初始化了被裝飾的函數(shù),__call__方法實(shí)現(xiàn)了在原有函數(shù)前后添加額外邏輯的功能。在調(diào)用say_hello函數(shù)時(shí),實(shí)際上會(huì)調(diào)用my_decorator的__call__方法。
Python裝飾器的應(yīng)用是廣泛的,它可以幫助我們實(shí)現(xiàn)緩存、日志、權(quán)限控制等功能。Python自帶的functools庫還提供了一些實(shí)用的裝飾器,例如lru_cache、wraps等。在實(shí)際開發(fā)中,裝飾器是一個(gè)必不可少的工具。