Python裝飾器使用接口測試的步驟
寫接口case時,有時需要對cae做一些共性的操作,最典型的場景如:獲取case執(zhí)行時間、打印log等。
有沒有一種辦法來集中處理共性操作從而避免在每個case中都寫相同的代碼(如:每個case都需要寫相同的獲取執(zhí)行時間的代碼)呢?
Python的裝飾器可以做到這一點。
可以這樣理解裝飾器,裝飾器運用閉包對目標函數進行裝飾(目標函數作為閉包外部函數的引用),即在執(zhí)行目標函數之前、之后執(zhí)行一些指定的代碼來完成想要的業(yè)務邏輯。
概念看起來總是晦澀,直接上語法吧。
第一步,先看一下閉包的語法:
def outer(a): b = 10 # inner是內函數 def inner(): # 在內函數中 用到了外函數的臨時變量 print(a + b) # 外函數的返回值是內函數的引用 return inner
第二步,再來看一下閉包的裝飾器版本的語法:
和上面閉包的例子只有一個區(qū)別:閉包中外部函數的引用是一個整數a,而此時是一個函數的引用(函數的引用也就是函數名)。
# 裝飾器用到閉包原理:外函數內部定義了一個內函數,內函數使用外函數的局部變量,并且外函數返回了內函數的引用 def outer(target): # 裝飾器函數 傳入一個想對其裝飾的目標函數的 引用,將在內函數中使用。 b = 10 c = 1 d = 5 # inner是內函數 def inner(): print(b + c) # 外部函數傳入的參數target,就是希望裝飾的目標函數的引用 target() # 這里實際上執(zhí)行了目標函數,想對這個函數進行裝飾,所以在該函數執(zhí)行之前和之后進行一番操作,具體什么操作看業(yè)務邏輯 print(c + d) # 外函數的返回值是內函數的引用 return inner
注意:target只是函數的一個引用(引用指向函數在內存中的位置),不會執(zhí)行。帶()時( target() )才會執(zhí)行該函數。
最后一步,再看一下裝飾器的語法:
@decorator def test_01():
所以可以總結出:裝飾器decorator是閉包的外部函數,即 outer() (裝飾器是一個函數,即閉包的外部函數),被裝飾函數test_01是閉包傳入的參數,即target。
舉個例子:
以統計各接口請求耗時為例。
裝飾器(decorat.py):
import time def time_consume(func): def inner(): time_start = time.time() # 目標函數開始之前取一下時間 print("\n接口請求前的時間是", time_start) func() time_end = time.time() # 目標函數結束之后取一下時間 print("接口請求后的時間是", time_end) t = time_end - time_start # 計算目標函數執(zhí)行花了多長時間 print("接口耗時:", t) return inner
接口(test_case.py):
import requests import decorat @decorat.time_consume def test_demo(): res = requests.get("https://www.baidu.com") assert res.status_code == 200
效果:
再來總結下這個例子的整個過程:
'''
@decorat.time_consume實際上執(zhí)行的是: test_demo = @decorat.time_consume(test_demo)
因為編程語言都是從右向左來解析執(zhí)行的,那么這句代碼會發(fā)生的事情是:1 、把目標函數test_demo(是一個變量名,里面存的是目標函數的引用) 傳入time_consume函數,被參數func接收,這時func也是目標函數的引用 func和test_demo指向同一個函數對象
2 、time_consume函數定義了內部函數inner,在inner里調用func,
這用到閉包的原理(閉包原理:外函數結束的時會把自身的引用綁定給內函數),外函數結束的時候會把func綁定給內函數,供內函數來使用
3、 外函數結束的時候把自己創(chuàng)建的內函數的引用inner返回給test_demo接收,
這時test_demo已經不是原來編寫的目標函數了,test_demo可以理解成是一個inner函數的實例對象,再執(zhí)行test_demo() 的時候實際上執(zhí)行了inner()的一個對象
4、 再執(zhí)行test_demo() 的時候 實際上執(zhí)行了inner() :
先執(zhí)行取時間,打印
之后執(zhí)行func(),才是執(zhí)行目標函數,即執(zhí)行test_demo()本身
最后再次取時間,打印結果
大白話版本:
其實就一句話:
被裝飾函數作為裝飾器外部函數的參數傳入,在裝飾器的內部函數中執(zhí)行被裝飾函數,并外加其他的代碼片段A,
這樣被裝飾函數除了具備自身的邏輯外,也擁有了裝飾器內部函數中代碼片段A的邏輯。使得無需修改被裝飾函數,
就增強了被裝飾函數的功能。
再來看兩種情景。
第一個:被裝飾函數有參數
一般接口測試的test_case不會想上面例子中提到的是一個函數,而是作為一個類的方法出現的,比如:
運行報錯了,報錯日志的意思是inner()需要0個入參,但是被傳入了1個。通過該報錯證明了上面提到的這個結論
原因是test_demo()有參數self,而inner()沒有定義入參。怎么解決呢,給inner()定義一個可變入參?先來看第二個問題,最后一起來證明我們的推測吧。
第二個:被裝飾函數有返回值
問題出現了,返回值打印出來是None,因為inner()里沒有變量去接收test_demo的返回值并返回嗎?帶著第一個問題的推測,一起來改下代碼。
做2處改動:
1、inner()定義可變入參
2、inner()里定義變量去接收test_demo的返回值并return該變量
test_case作為一個類的方法出現的問題解決了。
返回值也能被正常打印了。
改動后的裝飾器可以作為一個定義裝飾器的通用模板,基本可以給各種各樣的函數來裝飾了。
def decorat_demo(func): def inner(*args, **kwargs): # inner()接收可變參數 # any code before # 定義目標函數前的操作 # 調用目標函數 res = func(*args, **kwargs) # 定義變量接收目標函數返回值 # any code after # 定義目標函數后的操作 return res # 返回目標函數返回值 return inner
到此這篇關于Python裝飾器使用接口測試的步驟的文章就介紹到這了,更多相關python裝飾器接口測試內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
python之pyqt5通過按鈕改變Label的背景顏色方法
今天小編就為大家分享一篇python之pyqt5通過按鈕改變Label的背景顏色方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-06-06Python中函數eval和ast.literal_eval的區(qū)別詳解
eval函數在Python中做數據類型的轉換還是很有用的。它的作用就是把數據還原成它本身或者是能夠轉化成的數據類型。那么eval和ast.literal_val()的區(qū)別是什么呢?本文將大家介紹關于Python中函數eval和ast.literal_eval區(qū)別的相關資料,需要的朋友可以參考下。2017-08-08python使用response.read()接收json數據的實例
今天小編就為大家分享一篇python使用response.read()接收json數據的實例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-12-12macbook安裝環(huán)境chatglm2-6b的詳細過程
這篇文章主要介紹了macbook安裝chatglm2-6b的過程詳解,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-07-07