<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
寫介面case時,有時需要對cae做一些共性的操作,最典型的場景如:獲取case執行時間、列印log等。
有沒有一種辦法來集中處理共性操作從而避免在每個case中都寫相同的程式碼(如:每個case都需要寫相同的獲取執行時間的程式碼)呢?
Python的裝飾器可以做到這一點。
可以這樣理解裝飾器,裝飾器運用閉包對目標函數進行裝飾(目標函數作為閉包外部函數的參照),即在執行目標函數之前、之後執行一些指定的程式碼來完成想要的業務邏輯。
概念看起來總是晦澀,直接上語法吧。
def outer(a): b = 10 # inner是內函數 def inner(): # 在內函數中 用到了外函數的臨時變數 print(a + b) # 外函數的返回值是內函數的參照 return inner
和上面閉包的例子只有一個區別:閉包中外部函數的參照是一個整數a,而此時是一個函數的參照(函數的參照也就是函數名)。
# 裝飾器用到閉包原理:外函數內部定義了一個內函數,內函數使用外函數的區域性變數,並且外函數返回了內函數的參照 def outer(target): # 裝飾器函數 傳入一個想對其裝飾的目標函數的 參照,將在內函數中使用。 b = 10 c = 1 d = 5 # inner是內函數 def inner(): print(b + c) # 外部函數傳入的引數target,就是希望裝飾的目標函數的參照 target() # 這裡實際上執行了目標函數,想對這個函數進行裝飾,所以在該函數執行之前和之後進行一番操作,具體什麼操作看業務邏輯 print(c + d) # 外函數的返回值是內函數的參照 return inner
注意:target只是函數的一個參照(參照指向函數在記憶體中的位置),不會執行。帶()時( target() )才會執行該函數。
@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 # 計算目標函數執行花了多長時間 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實際上執行的是: test_demo = @decorat.time_consume(test_demo)
因為程式語言都是從右向左來解析執行的,那麼這句程式碼會發生的事情是:1 、把目標函數test_demo(是一個變數名,裡面存的是目標函數的參照) 傳入time_consume函數,被引數func接收,這時func也是目標函數的參照 func和test_demo指向同一個函數物件
2 、time_consume函數定義了內部函數inner,在inner裡呼叫func,
這用到閉包的原理(閉包原理:外函數結束的時會把自身的參照系結給內函數),外函數結束的時候會把func繫結給內函數,供內函數來使用
3、 外函數結束的時候把自己建立的內函數的參照inner返回給test_demo接收,
這時test_demo已經不是原來編寫的目標函數了,test_demo可以理解成是一個inner函數的範例物件,再執行test_demo() 的時候實際上執行了inner()的一個物件
4、 再執行test_demo() 的時候 實際上執行了inner() :
先執行取時間,列印
之後執行func(),才是執行目標函數,即執行test_demo()本身
最後再次取時間,列印結果
大白話版本:
其實就一句話:
被裝飾函數作為裝飾器外部函數的引數傳入,在裝飾器的內部函數中執行被裝飾函數,並外加其他的程式碼片段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裝飾器介面測試內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45