詳解Python裝飾器的四種定義形式
前言
裝飾器(decorator)在Python框架中扮演著重要角色,是Python中實(shí)現(xiàn)切面編程(AOP)的重要手段。
aspect-oriented programming (AOP) ,在不改變代碼自身的前提下增加程序功能
不改變代碼自身,但需要在函數(shù)和類頭上加一個(gè)標(biāo)注(annotation),這個(gè)標(biāo)注在Python里叫裝飾器,在java里叫注解。
在Python里,一共有四種組合形式。下面一一舉例。
用函數(shù)裝飾函數(shù)
采用一個(gè)函數(shù)定義裝飾器:
def decorate(f): def wrapper(*args): return f(*args)*2 return wrapper
然后作用在一個(gè)函數(shù)上:
@decorate def add(a, b): return a + b
測(cè)試一下效果:
def test_decorate(): sum = add(3, 5) assert sum == 16
用函數(shù)裝飾一個(gè)類
這里通過(guò)裝飾器實(shí)現(xiàn)單例模式:
def singleton(cls): instances = {} def wrapper(*args, **kwargs): if cls not in instances: instances[cls] = cls(*args, **kwargs) return instances[cls] return wrapper
使用該裝飾器:
@singleton class MyClass: def method(self): pass
于是,當(dāng)你定義多個(gè)對(duì)象時(shí),返回的是同一實(shí)例:
obj = MyClass() # creates a new instance obj2 = MyClass() # returns the same instance obj3 = MyClass() # returns the same instance ...
用類定義裝飾器,然后裝飾一個(gè)函數(shù)
先采用類定義一個(gè)裝飾器:
class Star: def __init__(self, n): self.n = n def __call__(self, fn): @wraps(fn) def wrapper(*args, **kwargs): result = fn(*args, **kwargs) return result return wrapper
再作用在一個(gè)函數(shù)上:
@Star(5) def add(a, b): return a + b
主要是在類中實(shí)現(xiàn)__call__方法。上面例子也可以簡(jiǎn)化:
class MyDecorator: def __init__(self, function): self.function = function def __call__(self, *args, **kwargs): # We can add some code # before function call self.function(*args, **kwargs) # We can also add some code # after function call. # adding class decorator to the function @MyDecorator def function(name, message ='Hello'): print("{}, {}".format(message, name))
用類定義裝飾器,然后裝飾一個(gè)類
先定義裝飾器:
class MyClassDecorator(object): _instances = dict() def __init__(self, name): pass def __call__(self, cls): class WrappedClass(cls): def say_hello(self): print(f'Hello: {self.username}') return WrappedClass
該裝飾器給被裝飾的類上添加了一個(gè)方法,名稱為say_hello()。使用如下:
@MyClassDecorator('test') class MyClass(): def __init__(self, username): self.username = username
然后:
def test_decoratorforclass(): obj = MyClass('user1') obj.say_hello()
打印出: Hello: user1
小結(jié)
學(xué)習(xí)類裝飾,對(duì)Python的內(nèi)部機(jī)制會(huì)有更多的了解。如__init__, call, __new__等內(nèi)置方法。
到此這篇關(guān)于Python裝飾器的四種定義形式的文章就介紹到這了,更多相關(guān)Python裝飾器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python生成13位或16位時(shí)間戳以及反向解析時(shí)間戳的實(shí)例
這篇文章主要介紹了python生成13位或16位時(shí)間戳以及反向解析時(shí)間戳的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03python函數(shù)不定長(zhǎng)參數(shù)使用方法解析
這篇文章主要介紹了python函數(shù)不定長(zhǎng)參數(shù)使用方法解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-12-12python pandas實(shí)現(xiàn)excel轉(zhuǎn)為html格式的方法
今天小編就為大家分享一篇python pandas實(shí)現(xiàn)excel轉(zhuǎn)為html格式的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-10-10在Qt5和PyQt5中設(shè)置支持高分辨率屏幕自適應(yīng)的方法
今天小編就為大家分享一篇在Qt5和PyQt5中設(shè)置支持高分辨率屏幕自適應(yīng)的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-06-06python BitMap算法處理20億隨機(jī)整數(shù)去重
這篇文章主要為大家介紹了python BitMap算法處理20億隨機(jī)整數(shù)去重,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01Python?Matplotlib繪制箱線圖boxplot()函數(shù)詳解
箱線圖一般用來(lái)展現(xiàn)數(shù)據(jù)的分布(如上下四分位值、中位數(shù)等),同時(shí)也可以用箱線圖來(lái)反映數(shù)據(jù)的異常情況,下面這篇文章主要給大家介紹了關(guān)于Python?Matplotlib繪制箱線圖boxplot()函數(shù)的相關(guān)資料,需要的朋友可以參考下2022-07-07在Django中管理Users和Permissions以及Groups的方法
這篇文章主要介紹了在Django中管理Users和Permissions以及Groups的方法,Django是最具人氣的Python web開發(fā)框架,需要的朋友可以參考下2015-07-07