0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

python學(xué)習(xí):三個(gè)測試庫的裝飾器實(shí)現(xiàn)思路

454398 ? 來源:Python貓公眾號(hào) ? 作者:豌豆花下貓 ? 2020-09-27 11:44 ? 次閱讀

Python 中實(shí)現(xiàn)參數(shù)化測試的幾個(gè)庫,并留下一個(gè)問題:

它們是如何做到把一個(gè)方法變成多個(gè)方法,并且將每個(gè)方法與相應(yīng)的參數(shù)綁定起來的呢?

我們再提煉一下,原問題等于是:在一個(gè)類中,如何使用裝飾器把一個(gè)類方法變成多個(gè)類方法(或者產(chǎn)生類似的效果)?

# 帶有一個(gè)方法的測試類
class TestClass:
    def test_func(self):
        pass

# 使用裝飾器,生成多個(gè)類方法
class TestClass:
    def test_func1(self):
        pass
    def test_func2(self):
        pass
    def test_func3(self):
        pass

Python 中裝飾器的本質(zhì)就是移花接木,用一個(gè)新的方法來替代被裝飾的方法。在實(shí)現(xiàn)參數(shù)化的過程中,我們介紹過的幾個(gè)庫到底用了什么手段/秘密武器呢?

1、ddt 如何實(shí)現(xiàn)參數(shù)化?

先回顧一下上篇文章中 ddt 庫的寫法:

import unittest
from ddt import ddt,data,unpack
@ddt
class MyTest(unittest.TestCase):
    @data((3, 1), (-1, 0), (1.2, 1.0))
    @unpack
    def test(self, first, second):
        pass

ddt 可提供 4 個(gè)裝飾器:1 個(gè)加在類上的 @ddt,還有 3 個(gè)加在類方法上的 @data、@unpack 和 @file_data(前文未提及)。

先看看加在類方法上的三個(gè)裝飾器的作用:

# ddt 版本(win):1.2.1
def data(*values):
    global index_len
    index_len = len(str(len(values)))
    return idata(values)

def idata(iterable):
    def wrapper(func):
        setattr(func, DATA_ATTR, iterable)
        return func
    return wrapper

def unpack(func):
    setattr(func, UNPACK_ATTR, True)
    return func

def file_data(value):
    def wrapper(func):
        setattr(func, FILE_ATTR, value)
        return func
    return wrapper

它們的共同作用是在類方法上 setattr() 添加屬性。至于這些屬性在什么時(shí)候使用?下面看看加在類上的 @ddt 裝飾器源碼:

第一層 for 循環(huán)遍歷了所有的類方法,然后是 if/elif 兩條分支,分別對(duì)應(yīng) DATA_ATTR/FILE_ATTR,即對(duì)應(yīng)參數(shù)的兩種來源:數(shù)據(jù)(@data)和文件(@file_data)。

elif 分支有解析文件的邏輯,之后跟處理數(shù)據(jù)相似,所以我們把它略過,主要看前面的 if 分支。這部分的邏輯很清晰,主要完成的任務(wù)如下:
? 遍歷類方法的參數(shù)鍵值對(duì)
? 根據(jù)原方法及參數(shù)對(duì),創(chuàng)建新的方法名
? 獲取原方法的文檔字符串
? 對(duì)元組和列表類型的參數(shù)作解包
? 在測試類上添加新的測試方法,并綁定參數(shù)與文檔字符串

分析源碼,可以看出,@data、@unpack 和 @file_data 這三個(gè)裝飾器主要是設(shè)置屬性并傳參,而 @ddt 裝飾器才是核心的處理邏輯。

這種將裝飾器分散(分別加在類與類方法上),再組合使用的方案,很不優(yōu)雅。為什么就不能統(tǒng)一起來使用呢?后面我們會(huì)分析它的難言之隱,先按下不表,看看其它的實(shí)現(xiàn)方案是怎樣的?

2、parameterized 如何實(shí)現(xiàn)參數(shù)化?

先回顧一下上篇文章中 parameterized 庫的寫法:

import unittest
from parameterized import parameterized
class MyTest(unittest.TestCase):
    @parameterized.expand([(3,1), (-1,0), (1.5,1.0)])
    def test_values(self, first, second):
        self.assertTrue(first > second)

它提供了一個(gè)裝飾器類 @parameterized,源碼如下(版本 0.7.1),主要做了一些初始的校驗(yàn)和參數(shù)解析,并非我們關(guān)注的重點(diǎn),略過。

我們主要關(guān)注這個(gè)裝飾器類的 expand() 方法,它的文檔注釋中寫到:

A "brute force" method of parameterizing test cases. Creates new test cases and injects them into the namespace that the wrapped function is being defined in. Useful for parameterizing tests in subclasses of 'UnitTest', where Nose test generators don't work.

關(guān)鍵的兩個(gè)動(dòng)作是:“creates new test cases(創(chuàng)建新的測試單元)”和“inject them into the namespace…(注入到原方法的命名空間)”。

關(guān)于第一點(diǎn),它跟 ddt 是相似的,只是一些命名風(fēng)格上的差異,以及參數(shù)的解析及綁定不同,不值得太關(guān)注。

最不同的則是,怎么令新的測試方法生效?

parameterized 使用的是一種“注入”的方式:

inspect 是個(gè)功能強(qiáng)大的標(biāo)準(zhǔn)庫,在此用于獲取程序調(diào)用棧的信息。前三句代碼的目的是取出 f_locals,它的含義是“l(fā)ocal namespace seen by this frame”,此處 f_locals 指的就是類的局部命名空間。

說到局部命名空間,你可能會(huì)想到 locals(),但是,我們之前有文章提到過“l(fā)ocals() 與 globals() 的讀寫問題”,locals() 是可讀不可寫的,所以這段代碼才用了 f_locals。

3、pytest 如何實(shí)現(xiàn)參數(shù)化?

按慣例先看看上篇文章中的寫法:

import pytest
@pytest.mark.parametrize("first,second", [(3,1), (-1,0), (1.5,1.0)])
def test_values(first, second):
    assert(first > second)

首先看到“mark”,pytest 里內(nèi)置了一些標(biāo)簽,例如 parametrize、timeout、skipif、xfail、tryfirst、trylast 等,還支持用戶自定義的標(biāo)簽,可以設(shè)置執(zhí)行條件、分組篩選執(zhí)行,以及修改原測試行為等等。

用法也是非常簡單的,然而,其源碼可復(fù)雜多了。我們這里只關(guān)注 parametrize,先看看核心的一段代碼:

根據(jù)傳入的參數(shù)對(duì),它復(fù)制了原測試方法的調(diào)用信息,存入待調(diào)用的列表里。跟前面分析的兩個(gè)庫不同,它并沒有在此創(chuàng)建新的測試方法,而是復(fù)用了已有的方法。在 parametrize() 所屬的 Metafunc 類往上查找,可以追蹤到 _calls 列表的使用位置:

最終是在 Function 類中執(zhí)行:

好玩的是,在這里我們可以看到幾行神注釋……

閱讀(粗淺涉獵) pytest 的源碼,真的是自討苦吃……不過,依稀大致可以看出,它在實(shí)現(xiàn)參數(shù)化時(shí),使用的是生成器的方案,遍歷一個(gè)參數(shù)則調(diào)用一次測試方法,而前面的 ddt 和 parameterized 則是一次性把所有參數(shù)解析完,生成 n 個(gè)新的測試方法,再交給測試框架去調(diào)度。

對(duì)比一下,前兩個(gè)庫的思路很清晰,而且由于其設(shè)計(jì)單純是為了實(shí)現(xiàn)參數(shù)化,不像 pytest 有什么標(biāo)記和過多的抽象設(shè)計(jì),所以更易讀易懂。前兩個(gè)庫發(fā)揮了 Python 的動(dòng)態(tài)特性,設(shè)置類屬性或者注入局部命名空間,而 pytest 倒像是從什么靜態(tài)語言中借鑒的思路,略顯笨拙。

4、最后小結(jié)

回到標(biāo)題中的問題“如何將一個(gè)方法變?yōu)槎鄠€(gè)方法?”除了在參數(shù)化測試中,不知還有哪些場景會(huì)有此訴求?歡迎留言討論。

本文分析了三個(gè)測試庫的裝飾器實(shí)現(xiàn)思路,通過閱讀源碼,我們可以發(fā)現(xiàn)它們各有千秋,這個(gè)發(fā)現(xiàn)本身還挺有意思。在使用裝飾器時(shí),表面看它們差異不大,但是真功夫的細(xì)節(jié)都隱藏在底下。

源碼分析的意義在于探究其所以然,在這次探究之旅中,讀者們可有什么收獲???

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • Function
    +關(guān)注

    關(guān)注

    0

    文章

    14

    瀏覽量

    9917
  • python
    +關(guān)注

    關(guān)注

    56

    文章

    4797

    瀏覽量

    84727
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    個(gè)月速成python+OpenCV圖像處理

    適用于哪些場景,然后通過Python編寫代碼來實(shí)現(xiàn)這些算法,并應(yīng)用于實(shí)際項(xiàng)目中,實(shí)現(xiàn)圖像的檢測、識(shí)別、分類、定位、測量等目標(biāo)。本文將介紹一個(gè)高效學(xué)習(xí)
    的頭像 發(fā)表于 11-29 18:27 ?141次閱讀
    一<b class='flag-5'>個(gè)</b>月速成<b class='flag-5'>python</b>+OpenCV圖像處理

    如何使用Python實(shí)現(xiàn)PID控制

    PID控制(比例-積分-微分控制)是一種常見的反饋控制算法,廣泛應(yīng)用于工業(yè)控制系統(tǒng)中。在Python實(shí)現(xiàn)PID控制,我們可以遵循以下步驟: 1. 理解PID控制原理 PID控制三個(gè)
    的頭像 發(fā)表于 11-14 09:09 ?504次閱讀

    Python解析:通過實(shí)現(xiàn)代理請求與數(shù)據(jù)抓取

    Python中,有多個(gè)可以幫助你實(shí)現(xiàn)代理請求和數(shù)據(jù)抓取。這些提供了豐富的功能和靈活的API,使得你可以輕松地發(fā)送HTTP請求、處理響應(yīng)、解析HTML/XML/JSON數(shù)據(jù),以及進(jìn)
    的頭像 發(fā)表于 10-24 07:54 ?175次閱讀

    使用CLB擴(kuò)展PWM實(shí)現(xiàn)F280049C驅(qū)動(dòng)三個(gè)電機(jī)

    電子發(fā)燒友網(wǎng)站提供《使用CLB擴(kuò)展PWM實(shí)現(xiàn)F280049C驅(qū)動(dòng)三個(gè)電機(jī).pdf》資料免費(fèi)下載
    發(fā)表于 09-27 10:13 ?1次下載
    使用CLB擴(kuò)展PWM<b class='flag-5'>實(shí)現(xiàn)</b>F280049C驅(qū)動(dòng)<b class='flag-5'>三個(gè)</b>電機(jī)

    【每天學(xué)點(diǎn)AI】一個(gè)例子帶你了解Python裝飾到底在干嘛!

    進(jìn)行“加料”呢?Python裝飾提供了一個(gè)更為優(yōu)雅的方式來增強(qiáng)現(xiàn)有函數(shù)的行為,并且不需要修改現(xiàn)有的函數(shù)代碼及調(diào)用方式。接下來通過一個(gè)案例來
    的頭像 發(fā)表于 09-20 16:54 ?560次閱讀
    【每天學(xué)點(diǎn)AI】一<b class='flag-5'>個(gè)</b>例子帶你了解<b class='flag-5'>Python</b><b class='flag-5'>裝飾</b><b class='flag-5'>器</b>到底在干嘛!

    基本理想電路元件的三個(gè)特征是什么

    基本理想電路元件是構(gòu)成電路的基本單元,它們具有三個(gè)基本特征:電壓-電流關(guān)系、能量轉(zhuǎn)換和電路參數(shù)。以下是對(duì)這三個(gè)特征的分析: 電壓-電流關(guān)系 理想電路元件的電壓-電流關(guān)系是其最基本的特征之一。這種關(guān)系
    的頭像 發(fā)表于 08-25 09:38 ?1163次閱讀

    pytorch和python的關(guān)系是什么

    在當(dāng)今的人工智能領(lǐng)域,Python已經(jīng)成為了最受歡迎的編程語言之一。Python的易學(xué)易用、豐富的和框架以及強(qiáng)大的社區(qū)支持,使其成為了數(shù)據(jù)科學(xué)、機(jī)器學(xué)習(xí)和深度
    的頭像 發(fā)表于 08-01 15:27 ?1984次閱讀

    可調(diào)變阻器三個(gè)引腳怎么區(qū)分

    可調(diào)變阻器,也稱為電位或可變電阻,是一種可以調(diào)節(jié)電阻值的電子元件。它廣泛應(yīng)用于各種電子設(shè)備中,如音頻設(shè)備、電源管理、電機(jī)控制等。可調(diào)變阻器有三個(gè)引腳,分別是固定端、滑動(dòng)端和可調(diào)端。這三個(gè)
    的頭像 發(fā)表于 07-24 11:12 ?1317次閱讀

    如何實(shí)現(xiàn)Python復(fù)制文件操作

    Python 中有許多“開蓋即食”的模塊(比如 os,subprocess 和 shutil)以支持文件 I/O 操作。在這篇文章中,你將會(huì)看到一些用 Python 實(shí)現(xiàn)文件復(fù)制的特殊方法。下面我們開始
    的頭像 發(fā)表于 07-18 14:53 ?426次閱讀

    opencv-python和opencv一樣嗎

    不一樣。OpenCV(Open Source Computer Vision Library)是一個(gè)開源的計(jì)算機(jī)視覺和機(jī)器學(xué)習(xí)軟件,它提供了大量的圖像和視頻處理功能。OpenCV-Pytho
    的頭像 發(fā)表于 07-16 10:38 ?1216次閱讀

    基于Python的深度學(xué)習(xí)人臉識(shí)別方法

    基于Python的深度學(xué)習(xí)人臉識(shí)別方法是一個(gè)涉及多個(gè)技術(shù)領(lǐng)域的復(fù)雜話題,包括計(jì)算機(jī)視覺、深度學(xué)習(xí)、以及圖像處理等。在這里,我將概述一個(gè)基本的
    的頭像 發(fā)表于 07-14 11:52 ?1274次閱讀

    深度學(xué)習(xí)常用的Python

    深度學(xué)習(xí)作為人工智能的一個(gè)重要分支,通過模擬人類大腦中的神經(jīng)網(wǎng)絡(luò)來解決復(fù)雜問題。Python作為一種流行的編程語言,憑借其簡潔的語法和豐富的支持,成為了深度
    的頭像 發(fā)表于 07-03 16:04 ?657次閱讀

    個(gè)籬笆三個(gè)樁——記晶體極管的發(fā)明

    個(gè)籬笆三個(gè)樁——記晶體極管的發(fā)明
    的頭像 發(fā)表于 05-12 08:14 ?756次閱讀
    一<b class='flag-5'>個(gè)</b>籬笆<b class='flag-5'>三個(gè)</b>樁——記晶體<b class='flag-5'>三</b>極管的發(fā)明

    Python自動(dòng)化測試框架及其應(yīng)用

    Pytest是一個(gè)非常成熟的全功能的Python測試框架,與python自帶的unittest測試框架類似,但是比unittest框架使用起
    的頭像 發(fā)表于 04-03 16:15 ?555次閱讀
    <b class='flag-5'>Python</b>自動(dòng)化<b class='flag-5'>測試</b>框架及其應(yīng)用

    如何使用Python進(jìn)行圖像識(shí)別的自動(dòng)學(xué)習(xí)自動(dòng)訓(xùn)練?

    如何使用Python進(jìn)行圖像識(shí)別的自動(dòng)學(xué)習(xí)自動(dòng)訓(xùn)練? 使用Python進(jìn)行圖像識(shí)別的自動(dòng)學(xué)習(xí)和自動(dòng)訓(xùn)練需要掌握一些重要的概念和技術(shù)。在本文中,我們將介紹如何使用
    的頭像 發(fā)表于 01-12 16:06 ?594次閱讀