日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網站首頁 編程語言 正文

Python中的裝飾器使用_python

作者:子休_ ? 更新時間: 2023-01-18 編程語言

Python裝飾器用法

Python的裝飾器是個好東西,它能干很多事情。

但對于新手,它看起來似乎沒那么簡單。

但事實上,裝飾器本身也只是個函數。

import time
def log(func):
? ? def wrapper(*args, **kwargs):
? ? ? ? start = time.time()
? ? ? ? result = func(*args, **kwargs)
? ? ? ? end = time.time()
? ? ? ? print("The func '{}' used {}s.".format(func.__name__, end-start))
? ? ? ? return result
? ? return warpper

這一個裝飾器,當我們這樣使用時

@log
def fucok(name):
? ? """Fucok someone"""
? ? print("Fucok", name)

它只是執行了fucok = log(fucok)這樣一句代碼而已。

也就是說,我們表面上是用fucok("myself"),事實上執行的都是log(fucok)("myself")。因為Python里面都是對象嘛。

同樣的道理,假設我們定義了一個帶參數的裝飾器logging,它實際上執行的是

func = logging(arguments)(func)

也就是上面那個不帶參數的裝飾器多定義一層就行了。

import time
def logging(arguments):
? ? def log(func):
? ? ? ? def warpper(*args, **kwargs):
? ? ? ? ? ? start = time.time()
? ? ? ? ? ? result = func(*args, **kwargs)
? ? ? ? ? ? end = time.time()
? ? ? ? ? ? print("The func '{}' used {}s.".format(func.__name__, end-start))
? ? ? ? ? ? return result
? ? ? ? return warpper
? ? # do something
? ? return log

但,當我們使用一個裝飾器之后,它會將原本的函數元信息給覆蓋掉。譬如:函數名稱,函數文檔等等。

例如上例

print(fucok.__name__)
print(fucok.__doc__)

你會發現,函數信息全部沒了!fucok它不叫fucok,改名叫wrapper了。它的文檔也變成了none。

解決辦法很簡單,定義裝飾器的時候用warps裝飾器裝飾接受原函數參數的那一層就行了。

這個來自functools模塊的裝飾器能幫你復制函數的元信息到被綁定的函數身上。

修改裝飾器如下(其實就加了一行代碼hhh)

import time
from functools import wraps
def log(func):
? ? @wraps(func)
? ? def warpper(*args, **kwargs):
? ? ? ? start = time.time()
? ? ? ? result = func(*args, **kwargs)
? ? ? ? end = time.time()
? ? ? ? print("The func '{}' used {}s.".format(func.__name__, end-start))
? ? ? ? return result
? ? return warpper

當我們再運行

print(fucok.__name__)
print(fucok.__doc__)

就能看到函數的的元信息沒變了。

裝飾器定義時加@wraps是個好習慣。

一個較為實用的裝飾器demo在該專題的另一篇文章:函數參數類型檢查

總結

原文鏈接:https://www.jianshu.com/p/4416e291c64d

欄目分類
最近更新