精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

python學習:三個測試庫的裝飾器實現思路

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

Python 中實現參數化測試的幾個庫,并留下一個問題:

它們是如何做到把一個方法變成多個方法,并且將每個方法與相應的參數綁定起來的呢?

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

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

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

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

1、ddt 如何實現參數化?

先回顧一下上篇文章中 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 個裝飾器:1 個加在類上的 @ddt,還有 3 個加在類方法上的 @data、@unpack 和 @file_data(前文未提及)。

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

# 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() 添加屬性。至于這些屬性在什么時候使用?下面看看加在類上的 @ddt 裝飾器源碼:

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

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

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

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

2、parameterized 如何實現參數化?

先回顧一下上篇文章中 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)

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

我們主要關注這個裝飾器類的 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.

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

關于第一點,它跟 ddt 是相似的,只是一些命名風格上的差異,以及參數的解析及綁定不同,不值得太關注。

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

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

inspect 是個功能強大的標準庫,在此用于獲取程序調用棧的信息。前三句代碼的目的是取出 f_locals,它的含義是“local namespace seen by this frame”,此處 f_locals 指的就是類的局部命名空間。

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

3、pytest 如何實現參數化?

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

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 里內置了一些標簽,例如 parametrize、timeout、skipif、xfail、tryfirst、trylast 等,還支持用戶自定義的標簽,可以設置執行條件、分組篩選執行,以及修改原測試行為等等。

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

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

最終是在 Function 類中執行:

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

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

對比一下,前兩個庫的思路很清晰,而且由于其設計單純是為了實現參數化,不像 pytest 有什么標記和過多的抽象設計,所以更易讀易懂。前兩個庫發揮了 Python 的動態特性,設置類屬性或者注入局部命名空間,而 pytest 倒像是從什么靜態語言中借鑒的思路,略顯笨拙。

4、最后小結

回到標題中的問題“如何將一個方法變為多個方法?”除了在參數化測試中,不知還有哪些場景會有此訴求?歡迎留言討論。

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

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

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Function
    +關注

    關注

    0

    文章

    14

    瀏覽量

    9865
  • python
    +關注

    關注

    53

    文章

    4747

    瀏覽量

    83926
收藏 人收藏

    評論

    相關推薦

    深度學習常用的Python

    深度學習作為人工智能的一重要分支,通過模擬人類大腦中的神經網絡來解決復雜問題。Python作為一種流行的編程語言,憑借其簡潔的語法和豐富的支持,成為了深度
    的頭像 發表于 07-03 16:04 ?227次閱讀

    如何使用Python進行圖像識別的自動學習自動訓練?

    如何使用Python進行圖像識別的自動學習自動訓練? 使用Python進行圖像識別的自動學習和自動訓練需要掌握一些重要的概念和技術。在本文中,我們將介紹如何使用
    的頭像 發表于 01-12 16:06 ?397次閱讀

    python有哪些

    Python 作為一門功能強大的編程語言,擁有豐富的第,幾乎覆蓋了各個領域的應用。下面是一些常見且廣泛應用的 Python
    的頭像 發表于 11-29 14:31 ?1798次閱讀

    python中如何引入math

    Python中,要使用math,首先需要先引入它。mathPython的一標準,它提
    的頭像 發表于 11-22 11:03 ?3080次閱讀

    python中if三個條件怎么寫

    確定程序的執行路徑。這時,我們可以使用 if-elif-else 結構來處理多個條件。 if-elif-else 結構是一被廣泛使用的控制結構,在許多編程語言中都有類似的實現。它允許我們依次測試多個條件,并根據條件的真假執行相
    的頭像 發表于 11-21 16:42 ?861次閱讀

    如何寫一簡單的裝飾

    今天介紹的是一已經存在十三年,但是依舊不紅的 decorator,好像很少有人知道他的存在一樣。 這個可以幫你做什么呢 ? 其實很簡單,就是可以幫你更方便地寫python
    的頭像 發表于 11-01 09:54 ?375次閱讀
    如何寫一<b class='flag-5'>個</b>簡單的<b class='flag-5'>裝飾</b><b class='flag-5'>器</b>

    Python 自制簡單實用的日志裝飾

    ,因為本質上裝飾就是一返回函數的“高階”函數而已: 1) 函數作為參數傳遞進裝飾。 2) 裝飾
    的頭像 發表于 10-31 15:05 ?408次閱讀
    <b class='flag-5'>Python</b> 自制簡單實用的日志<b class='flag-5'>裝飾</b><b class='flag-5'>器</b>

    Python自制簡單實用的日志裝飾

    ,因為本質上裝飾就是一返回函數的“高階”函數而已: 1) 函數作為參數傳遞進裝飾。 2) 裝飾
    的頭像 發表于 10-21 14:39 ?566次閱讀
    <b class='flag-5'>Python</b>自制簡單實用的日志<b class='flag-5'>裝飾</b><b class='flag-5'>器</b>

    Python 梯度計算模塊如何實現邏輯回歸模型

    的標準數據下編寫的損失函數,它就可以自動計算損失函數的導數(梯度)。 我們將從普通斜率計算開始,介紹到如何只使用它來實現邏輯回歸模型。 1.準備 開始之前,你要確保Python
    的頭像 發表于 10-21 11:01 ?401次閱讀
    <b class='flag-5'>Python</b> 梯度計算模塊如何<b class='flag-5'>實現</b>一<b class='flag-5'>個</b>邏輯回歸模型

    TinyDB :一Python編寫的輕量級數據

    TinyDB 是一Python 編寫的輕量級數據,一共只有1800行代碼,沒有外部依賴項。 TinyDB的目標是降低小型 Python 應用程序使用數據
    的頭像 發表于 10-21 10:22 ?581次閱讀

    Bulbea:用于股票市場預測和建模的Python

    Bulbea 是一基于深度學習開發的,用于股票市場預測和建模的Python。 Bulbea 自帶了不少可用于股票深度學習訓練及
    的頭像 發表于 10-17 11:01 ?363次閱讀
    Bulbea:用于股票市場預測和建模的<b class='flag-5'>Python</b><b class='flag-5'>庫</b>

    如何使用Python和PinPong控制Arduino

    與傳感和其他物理設備集成的應用程序。如果您已經掌握了Python的基礎知識,那么您可以通過使用Python來控制Arduino來入門。本文目的主要是向您展示如何使用PinPong
    的頭像 發表于 10-13 10:59 ?671次閱讀
    如何使用<b class='flag-5'>Python</b>和PinPong<b class='flag-5'>庫</b>控制Arduino

    python讀取數據數據 python查詢數據 python數據連接

    使用第,包括MySQLDB、sqlite3、psycopg2等。其中MySQLDB是Python連接MySQL數據的一
    的頭像 發表于 08-28 17:09 ?1549次閱讀

    python有什么用 如何用python創建數據

    開發、游戲開發、機器學習、自然語言處理等領域。 在本文中,我們將介紹如何使用Python創建數據。首先,讓我們了解一下什么是數據。 什么是數據
    的頭像 發表于 08-28 16:41 ?1024次閱讀

    SQLite數據python的區別

    、數據科學等方面。SQLite數據Python之間有很多不同之處,下面將詳細解析它們之間的區別。 1. 數據類型 SQLite是一種關系型數據管理系統(RDBMS),而
    的頭像 發表于 08-28 16:41 ?686次閱讀