python裝飾器底層原理詳解
1 python裝飾器的作用
被裝飾對象加上裝飾器(戴了個帽子),被裝飾對象獲得了更強(qiáng)大的功能。
2 python裝飾器的原理
- python裝飾器本身是一個函數(shù)
- 這個函數(shù)的參數(shù)是一個函數(shù)對象
- 這個函數(shù)的返回值也是一個函數(shù)對象,這個函數(shù)的功能更強(qiáng)
- 大python裝飾器是python的一個語法糖(更簡便的語法)
3 python裝飾器的實(shí)現(xiàn)
3.1 最簡陋的裝飾器
# 自定義裝飾器
def super_(func):
def wrapper():
print('把內(nèi)褲穿到外面來,變身超人')
func()
print('會飛!')
return wrapper
@super_
def man():
print('會走')
# @super_原始語法結(jié)構(gòu)
# man = super_(man)
man()運(yùn)行結(jié)果如下:

3.2 給有返回值的函數(shù)加上裝飾器
'''
給有返回值的函數(shù)加上裝飾器
'''
import time
def decorate(func):
def wrapper():
print('開始執(zhí)行時間:' + time.strftime('%Y-%m-%d %H:%M:%S'))
result = func()
print('結(jié)束執(zhí)行時間:' + time.strftime('%Y-%m-%d %H:%M:%S'))
return result
return wrapper
@decorate
def normal_func():
time.sleep(1)
print('normal_func執(zhí)行中......')
return 2 + 2
# @decorate原始語法結(jié)構(gòu)
# normal_func = decorate(normal_func)
print(normal_func())運(yùn)行結(jié)果如下:

3.3 給有返回值和參數(shù)的函數(shù)加上裝飾器
'''
給有返回值和參數(shù)的函數(shù)加上裝飾器
'''
import time
def decorate(func):
def wrapper(*args, **kwargs):
print('開始執(zhí)行時間:' + time.strftime('%Y-%m-%d %H:%M:%S'))
result = func(*args, **kwargs)
print('結(jié)束執(zhí)行時間:' + time.strftime('%Y-%m-%d %H:%M:%S'))
return result
return wrapper
@decorate
def normal_func1(a, b):
time.sleep(1)
print('normal_func1執(zhí)行中......')
return a + b
@decorate
def normal_func2(a, b, c):
time.sleep(1)
print('normal_func2執(zhí)行中......')
return a + b + c
# @decorate原始語法結(jié)構(gòu)
# normal_func1 = decorate(normal_func1)
# normal_func2 = decorate(normal_func2)
print(normal_func1(1, 2))
print(normal_func2(1, 2, 3))運(yùn)行結(jié)果如下:

可變參數(shù)原理:
1 定義時使用可變參數(shù):在函數(shù)定義時使用args,在函數(shù)調(diào)用時,所有未匹配到的位置參數(shù),會被放到args這個元組當(dāng)中。
在函數(shù)定義時使用**kwargs,在函數(shù)調(diào)用時,所有未匹配到的關(guān)鍵字參數(shù),會被放到kwargs這個字典當(dāng)中。
2 調(diào)用時使用可變參數(shù):在函數(shù)調(diào)用時使用args,是把a(bǔ)rgs這個元組解包,元組內(nèi)的每個元素作為函數(shù)的位置參數(shù)傳遞。
在函數(shù)調(diào)用時使用**kwargs,是把kwargs這個字典解包,字典內(nèi)的每個元素作為函數(shù)的關(guān)鍵字參數(shù)傳遞。
def test1(a, b, c, d):
print(a+b+c+d)
print('傳統(tǒng)調(diào)用'.center(60, '='))
test1(1, 2, 3, 4) # 位置參數(shù)
test1(b=2, c=3, d=4, a=1) # 關(guān)鍵字參數(shù)
# 在函數(shù)定義時使用*args,在函數(shù)調(diào)用時,所有未匹配到的位置參數(shù),會被放到args這個元組當(dāng)中
# 在函數(shù)定義時使用**kwargs,在函數(shù)調(diào)用時,所有未匹配到的關(guān)鍵字參數(shù),會被放到kwargs這個字典當(dāng)中
def test2(*args, **kwargs):
print(args)
print(type(args))
print(kwargs)
print(type(kwargs))
print('定義時使用可變參數(shù)'.center(60, '='))
test2(1, 2, 3, 4, b=5, c=6)
# 在函數(shù)調(diào)用時使用*args,是把a(bǔ)rgs這個元組解包,元組內(nèi)的每個元素作為函數(shù)的位置參數(shù)傳遞。
# 在函數(shù)調(diào)用時使用**kwargs,是把kwargs這個字典解包,字典內(nèi)的每個元素作為函數(shù)的關(guān)鍵字參數(shù)傳遞。
print('調(diào)用時使用可變參數(shù)'.center(60, '='))
test1(*(1, 2, 3, 4))
test1(**{'a':1, 'b':2, 'c':3, 'd':4})
test1(*(1, 2), **{'c':3, 'd':4})運(yùn)行結(jié)果如下:

3.4 讓我還是那個我
'''
讓我還是那個我
'''
import time
from functools import wraps
def decorate(func):
@wraps(func) # 把wrapper的內(nèi)置屬性轉(zhuǎn)換成func的內(nèi)置屬性(name/doc)
def wrapper(*args, **kwargs):
print('開始執(zhí)行時間:' + time.strftime('%Y-%m-%d %H:%M:%S'))
result = func(*args, **kwargs)
print('結(jié)束執(zhí)行時間:' + time.strftime('%Y-%m-%d %H:%M:%S'))
return result
# wrapper.__name__ = func.__name__
# wrapper.__doc__ = func.__doc__
return wrapper
@decorate
def normal_func(a, b):
'''
這是一個測試函數(shù)
'''
time.sleep(1)
print('normal_func1執(zhí)行中......')
return a + b
print(normal_func.__name__)
print(normal_func.__doc__)
4 python裝飾器在自動化測試框架中的應(yīng)用
4.1 從一個需求開始
對自動化測試腳本需要增加日志打印功能:
- 測試用例執(zhí)行前打?。?/strong>測試用例【xxx】開始執(zhí)行
- 測試用例執(zhí)行完打?。?/strong>測試用例【xxx】執(zhí)行完畢
- 測試用例執(zhí)行完打印:測試用例【xxx】執(zhí)行耗時:xx秒
- 測試用例執(zhí)行完打印分割線
import time
from functools import wraps
def log_decorator(func):
@wraps(func) # 把wrapper的內(nèi)置屬性轉(zhuǎn)換成func的內(nèi)置屬性(name/doc)
def wrapper(*args, **kwargs):
print('測試用例[{}]開始執(zhí)行'.format(func.__name__))
time_start = time.time()
result = func(*args, **kwargs)
time_end = time.time()
print('測試用例[{}]執(zhí)行完畢'.format(func.__name__))
print('測試用例[{}]執(zhí)行耗時:%.2f秒'.format(func.__name__) % (time_end - time_start))
print('分割線'.center(60, '='))
return result
return wrapper到此這篇關(guān)于python裝飾器底層原理講解的文章就介紹到這了,更多相關(guān)python裝飾器底層原理講解內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python實(shí)現(xiàn)遠(yuǎn)程控制電腦
這篇文章主要為大家詳細(xì)介紹了python實(shí)現(xiàn)遠(yuǎn)程控制電腦,具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-05-05
Django框架ORM操作數(shù)據(jù)庫不生效問題示例解決方法
本文詳細(xì)描述使用Django 的ORM框架操作PostgreSQL數(shù)據(jù)庫刪除不生效問題的定位過程及解決方案,并總結(jié)使用ORM框架操作數(shù)據(jù)庫不生效的問題的通用定位方法,感興趣的朋友跟隨小編一起看看吧2023-01-01
Python編程使用PyQt5庫實(shí)現(xiàn)動態(tài)水波進(jìn)度條示例
這篇文章主要介紹了Python編程使用PyQt5庫實(shí)現(xiàn)動態(tài)水波進(jìn)度條的示例代碼解析,有需要的朋友可以借鑒參考下希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪2021-10-10
python使用collections模塊的容器數(shù)據(jù)類型高效處理數(shù)據(jù)
這篇文章主要為大家介紹了python使用collections模塊的容器數(shù)據(jù)類型高效處理數(shù)據(jù)的方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06
python如何爬取網(wǎng)站數(shù)據(jù)并進(jìn)行數(shù)據(jù)可視化
這篇文章主要介紹了python爬取拉勾網(wǎng)數(shù)據(jù)并進(jìn)行數(shù)據(jù)可視化,爬取拉勾網(wǎng)關(guān)于python職位相關(guān)的數(shù)據(jù)信息,并將爬取的數(shù)據(jù)已csv各式存入文件,然后對csv文件相關(guān)字段的數(shù)據(jù)進(jìn)行清洗,并對數(shù)據(jù)可視化展示,包括柱狀圖展示、直方圖展示,需要的朋友可以參考下2019-07-07

