一起來看看python的裝飾器代碼
裝飾器通用模型
def wrapper(fn):
def inner(*args, **kwargs):
ret = fn(*args, **kwargs)
return ret
return inner
裝飾器幾個關(guān)鍵點
""" 1.函數(shù)可以當(dāng)參數(shù)傳遞 2.函數(shù)可以作為返回值進(jìn)行返回 3.函數(shù)名稱可以當(dāng)成變量一樣進(jìn)行賦值操作 裝飾器本質(zhì)上是個閉包 在不改變原有函數(shù)調(diào)用的情況下,給函數(shù)增加新的功能 """
舉個例子
#!/usr/bin/python
def admin(game):
def inner(*args, **kwargs): # inner添加了參數(shù),args 一定是個元組 kwargs 一定是字典
print('打開Wg')
result = game(*args, **kwargs) # * ** 表示把args元組和kwargs打散成位置參數(shù),關(guān)鍵字參數(shù)傳遞進(jìn)去
print('關(guān)閉Wg')
return result
return inner
@admin
def play_dnf(username, password):
print(f'開始玩DNF,賬號:{username},密碼:{password}')
print('刀斬肉身,心斬靈魂')
return '掉落:戮蠱的哀鳴炮'
@admin
def play_wow(race, occupation, server_name, camp):
print(f'開始玩魔獸世界,種族:{race},職業(yè):{occupation},服務(wù)器:{server_name},陣營:{camp}')
print('為了辛多雷的榮耀')
return '掉落:灰燼使者'
if __name__ == '__main__':
ret1 = play_dnf('大馬猴', '888888')
print(ret1)
ret2 = play_wow('血精靈', '圣騎士', '回音山', '部落')
print(ret2)
這代碼還是很好懂的,我就不解釋了,然后是執(zhí)行結(jié)果
python demo.py 打開Wg 開始玩DNF,賬號:大馬猴,密碼:888888 刀斬肉身,心斬靈魂 關(guān)閉Wg 掉落:戮蠱的哀鳴炮 打開Wg 開始玩魔獸世界,種族:血精靈,職業(yè):圣騎士,服務(wù)器:回音山,陣營:部落 為了辛多雷的榮耀 關(guān)閉Wg 掉落:灰燼使者 Process finished with exit code 0
一個函數(shù)被多個裝飾器裝飾,又將如何執(zhí)行呢?
#!/usr/bin/python
def wrapper1(fn):
def inner(*args, **kwargs):
print('這是w1進(jìn)入')
ret = fn(*args, **kwargs)
print('這是w1出去')
return ret
return inner
def wrapper2(fn):
def inner(*args, **kwargs):
print('這是w2進(jìn)入')
ret = fn(*args, **kwargs)
print('這是w2出去')
return ret
return inner
@wrapper1
@wrapper2
def target():
print('我是目標(biāo)')
if __name__ == '__main__':
target()
直接給出執(zhí)行順序
一個函數(shù)被多個裝飾器裝飾的執(zhí)行順序
# w1 w2 target w2 w1
帶參數(shù)的裝飾器
裝飾器的語法允許我們在調(diào)用時,提供其它參數(shù),比如@decorator(a)。這樣,就為裝飾器的編寫和使用提供了更大的靈活性。
(在上面又套了一層函數(shù))
比如,我們可以在裝飾器中指定日志的等級,因為不同業(yè)務(wù)函數(shù)可能需要的日志級別是不一樣的。
def use_logging(level):
def decorator(func):
def wrapper(*args, **kwargs):
if level == "warn":
logging.warn("%s is running" % func.__name__)
elif level == "info":
logging.info("%s is running" % func.__name__)
return func(*args)
return wrapper
return decorator
@use_logging(level="warn")
def foo(name='foo'):
print("i am %s" % name)
foo()
類裝飾器
沒錯,裝飾器不僅可以是函數(shù),還可以是類,相比函數(shù)裝飾器,類裝飾器具有靈活度大、高內(nèi)聚、封裝性等優(yōu)點。使用類裝飾器主要依靠類的__call__方法,當(dāng)使用 @ 形式將裝飾器附加到函數(shù)上時,就會調(diào)用此方法。
class Foo(object):
def __init__(self, func):
self._func = func
def __call__(self):
print ('class decorator runing')
self._func()
print ('class decorator ending')
@Foo
def bar():
print ('bar')
bar()
總結(jié)
本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Python實現(xiàn)系統(tǒng)交互(subprocess)
我們幾乎可以在任何操作系統(tǒng)上通過命令行指令與操作系統(tǒng)進(jìn)行交互,本文實現(xiàn)了Python系統(tǒng)交互,具有一定的參考價值,感興趣的可以了解一下2021-07-07
解決pymongo連接數(shù)據(jù)庫報錯certificate verify failed:certific
這篇文章主要介紹了解決pymongo連接數(shù)據(jù)庫報錯certificate verify failed:certificate has expired問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01
使用50行Python代碼從零開始實現(xiàn)一個AI平衡小游戲
本文會為大家展示機器學(xué)習(xí)專家 Mike Shi 如何用 50 行 Python 代碼創(chuàng)建一個 AI,使用增強學(xué)習(xí)技術(shù),玩耍一個保持桿子平衡的小游戲。本文給大家?guī)韺崿F(xiàn)思路及簡單代碼,感興趣的朋友跟隨小編一起看看吧2018-11-11
Python數(shù)據(jù)結(jié)構(gòu)與算法中的隊列詳解(1)
這篇文章主要為大家詳細(xì)介紹了Python的隊列,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-03-03

