国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

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

完善資料讓更多小伙伴認識你,還能領取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 循環(huán)遍歷了所有的類方法,然后是 if/elif 兩條分支,分別對應 DATA_ATTR/FILE_ATTR,即對應參數的兩種來源:數據(@data)和文件(@file_data)。

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

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

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

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(創(chuàng)建新的測試單元)”和“inject them into the namespace…(注入到原方法的命名空間)”。

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

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

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

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

說到局部命名空間,你可能會想到 locals(),但是,我們之前有文章提到過“l(fā)ocals() 與 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 等,還支持用戶自定義的標簽,可以設置執(zhí)行條件、分組篩選執(zhí)行,以及修改原測試行為等等。

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

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

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

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

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

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

4、最后小結

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

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

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

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

    關注

    0

    文章

    14

    瀏覽量

    10308
  • python
    +關注

    關注

    57

    文章

    4876

    瀏覽量

    90029
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    如何在 Vision Five 2 上安裝 python

    這可能是一完全愚蠢的問題,但我如何在 Vision Five 2 上安裝 python 。 使用該命令后,它給了我這個錯誤。 默認為用戶安裝,因為普通站點包不可寫 錯誤:找不到滿足要求
    發(fā)表于 03-06 07:51

    無法去除 Python VisionFive.i2c 的終端輸出?

    燒的官方最新八月份的 debian 12 的系統(tǒng)。 根據這個案例安裝好了 python 環(huán)境和 VisionFive 。 執(zhí)行下面這條代碼: import VisionFive.i2c
    發(fā)表于 02-25 06:13

    駿馬奔騰,芯向未來:SiC功率器件的“三個必然”與丙午馬年的產業(yè)躍遷

    駿馬奔騰,芯向未來:SiC功率器件的“三個必然”與丙午馬年的產業(yè)躍遷 日期: 2026年 丙午馬年 除夕 主題: 傾佳電子楊茜“三個必然”戰(zhàn)略論斷與基本半導體碳化硅技術的產業(yè)替代邏輯 關鍵詞
    的頭像 發(fā)表于 02-15 10:33 ?42次閱讀
    駿馬奔騰,芯向未來:SiC功率器件的“<b class='flag-5'>三個</b>必然”與丙午馬年的產業(yè)躍遷

    三個經典開關電源實際問題解析

    下面三個關于多路輸出電源連續(xù)工作模式的問題,測試一下您的電源設計知識。然后核對您的答案。
    的頭像 發(fā)表于 01-22 16:08 ?1882次閱讀
    <b class='flag-5'>三個</b>經典開關電源實際問題解析

    束管式光纜最簡單三個步驟

    束管式光纜的安裝或施工可簡化為以下三個核心步驟,確保光纖傳輸性能穩(wěn)定且操作高效: 第一步:路由規(guī)劃與準備 路徑勘察 確定光纜敷設路線,避開強電磁干擾源(如高壓線)、尖銳物體或易受外力破壞的區(qū)域(如
    的頭像 發(fā)表于 01-07 10:12 ?235次閱讀

    京東關鍵詞搜索商品列表的Python爬蟲實戰(zhàn)

    京東關鍵詞搜索商品列表 Python 爬蟲實戰(zhàn) 你想要實現京東關鍵詞搜索商品的爬蟲,我會從 合規(guī)聲明、環(huán)境準備、頁面分析、代碼實現、反爬優(yōu)化 五方面展開,幫助你完成實戰(zhàn)項目。 一、前
    的頭像 發(fā)表于 01-04 10:16 ?736次閱讀

    Python調用API教程

    不同系統(tǒng)之間的信息交互。在這篇文章中,我們將詳細介紹Python調用API的方法和技巧。 一、用Requests發(fā)送HTTP請求 使用Python調用API的第一步是發(fā)送HTTP
    的頭像 發(fā)表于 11-03 09:15 ?881次閱讀

    未來工業(yè)AI發(fā)展的三個必然階段

    與優(yōu)化 能力的深層革命。 未來十年,工業(yè)AI的發(fā)展將經歷三個清晰的階段:? 智能輔助 → 智能決策 → 自主優(yōu)化 。這次進化,構成了工業(yè)從“人控機器”到“機器共智”的核心路徑。 一、第一階段:智能輔助(AI for Assistance) 時間窗口:
    的頭像 發(fā)表于 10-27 15:47 ?452次閱讀
    未來工業(yè)AI發(fā)展的<b class='flag-5'>三個</b>必然階段

    termux如何搭建python游戲

    模擬,支持通過APT包管理安裝軟件。搭建Python游戲開發(fā)環(huán)境前需完成以下基礎配置: 1. 更換國內源 為提升下載速度,需替換Termux默認源為清華源,執(zhí)行以下命令: ```bash
    發(fā)表于 08-29 07:06

    怎么結合嵌入式,Linux,和FPGA三個方向達到一均衡發(fā)展?

    在嵌入式領域,不少人都懷揣著讓嵌入式、Linux 和 FPGA 三個方向實現均衡發(fā)展的夢想,然而實踐中卻面臨諸多挑戰(zhàn)。就像備受矚目的全棧工程師稚暉君,他從大學玩單片機起步,憑借將智能算法融入嵌入式而
    的頭像 發(fā)表于 06-25 10:08 ?863次閱讀
    怎么結合嵌入式,Linux,和FPGA<b class='flag-5'>三個</b>方向達到一<b class='flag-5'>個</b>均衡發(fā)展?

    18常用的強化學習算法整理:從基礎方法到高級模型的理論技術與代碼實現

    本來轉自:DeepHubIMBA本文系統(tǒng)講解從基本強化學習方法到高級技術(如PPO、A3C、PlaNet等)的實現原理與編碼過程,旨在通過理論結合代碼的方式,構建對強化學習算法的全面理解。為確保內容
    的頭像 發(fā)表于 04-23 13:22 ?1611次閱讀
    18<b class='flag-5'>個</b>常用的強化<b class='flag-5'>學習</b>算法整理:從基礎方法到高級模型的理論技術與代碼<b class='flag-5'>實現</b>

    python入門圣經-高清電子書(建議下載)

    和Pygal 等強大的Python 和工具介紹,以及列表、字典、if 語句、類、文件與異常、代碼測試等內容; 第二部分將理論付諸實踐,講解如何開發(fā)三個項目,包括簡單的
    發(fā)表于 04-10 16:53

    半導體制造過程中的三個主要階段

    前段工藝(Front-End)、中段工藝(Middle-End)和后段工藝(Back-End)是半導體制造過程中的三個主要階段,它們在制造過程中扮演著不同的角色。
    的頭像 發(fā)表于 03-28 09:47 ?7333次閱讀
    半導體制造過程中的<b class='flag-5'>三個</b>主要階段

    如何檢測極管的三個

    可以用萬用表來初步確定極管的好壞及類型 (NPN 型還是 PNP 型 ),并辨別出e(發(fā)射極)、b(基極)、c(集電極)三個電極。
    發(fā)表于 03-08 16:40

    如何區(qū)分場效應管mos管三個引腳

    場效應管mos管三個引腳怎么區(qū)分
    發(fā)表于 03-07 09:20 ?0次下載