Pytest框架 — 07、Pytest的Fixture(部分前後置)(二)

来源:https://www.cnblogs.com/qishuaiRisen/archive/2022/08/15/16589328.html
-Advertisement-
Play Games

4、Fixture的相互調用 示例: import pytest # 第一層fixture @pytest.fixture() def fixture_1(): data = "fixture_1" print("這是第一層fixture") return data # 第二層fixture @py ...


目錄

4、Fixture的相互調用

示例:

import pytest


# 第一層fixture
@pytest.fixture()
def fixture_1():
    data = "fixture_1"
    print("這是第一層fixture")
    return data


# 第二層fixture
@pytest.fixture()
def fixture_2(fixture_1):  # 在這裡傳遞調用第一層fixture
    print("這是第二層fixture")

# 第三層fixture
@pytest.fixture()
@pytest.mark.usefixtures("fixture_1")  # 這種fixture相互調用方式不生效
def fixture_3():
    print("這是第三層fixture")


# 測試類
class TestClass:
    def test_1(self, fixture_1):
        print("調用第1層fixture,返回值為{}".format(fixture_1))

    def test_2(self, fixture_2):
        print("調用第2層fixture,返回值為{}".format(fixture_2))

    def test_3(self, fixture_3):
        print("調用第3層fixture")


"""
執行結果
fixture/use_fixture_each_other.py::TestClass::test_1 這是第一層fixture
調用第1層fixture,返回值為fixture_1
PASSED
fixture/use_fixture_each_other.py::TestClass::test_2 這是第一層fixture
這是第二層fixture
調用第2層fixture,返回值為None
PASSED
fixture/use_fixture_each_other.py::TestClass::test_3 這是第三層fixture
調用第3層fixture
PASSED
"""

示例說明:

  • 在調用存在相互調用的fixture時,執行順序由頂層fixture逐層向下執行(如示例test_2優先執行fixture_1再執行fixture_2
  • 當調用存在相互調用的fixture時,直接被調用的fixture不會將自己接收到的返回值返回給調用方(如示例test_2不會接收到由fixture_1返回的值)
  • 相互調用fixture時不能使用pytest.mark.usefixture裝飾器(如示例fixture_3

5、Fixture復用

不同的測試函數可以請求相同的fixture,每個測試函數都會獲得該fixture的各自結果。不同的測試函數不互相影響,這樣可以保證每個測試函數都能得到乾凈一致的數據。
示例

import pytest


@pytest.fixture()
def fixture_1():
    return []


def test_1(fixture_1):
    fixture_1.append(0)
    print(f"結果:{fixture_1}")


def test_2(fixture_1):
    fixture_1.append(9)
    print(f"結果:{fixture_1}")

"""
執行結果
fixture/reuse_fixture.py::test_1 結果:[0]
PASSED
fixture/reuse_fixture.py::test_2 結果:[9]
PASSED
"""

示例說明
由上面例子可以看出,兩個測試函數都調用了fixture_1,但每次調用返回的結果是一樣的,並不會因為test_1fixture_1先添加了0而影響test_2調用的fixture_1返回的結果

6、Fixture緩存返回結果

同一個測試函數多次調用同一個fixture時,第一次執行該fixture函數之後,會把返回結果緩存起來,不會再次執行它們
示例:

import pytest


@pytest.fixture()
def fixture_1():
    print("執行fixture_1")
    return "a"

@pytest.fixture()
def fixture_2():
    print("執行fixture_2")
    return []

@pytest.fixture()
def fixture_append(fixture_1, fixture_2):
    print("執行fixture_append")
    fixture_2.append(fixture_1)

def test_fixture(fixture_append, fixture_2, fixture_1):
    print("執行測試函數")
    print(fixture_2)

"""
執行結果
fixture/fixture_cache.py::test_fixture 執行fixture_1
執行fixture_2
執行fixture_append
執行測試函數
['a']
PASSED

"""

示例說明:
由上面例子可以看出在fixture函數fixture_append中,fixture_1第一次被請求返回字母afixture_2被第二次被請求返回空列表,在測試函數test_fixturefixture_2第二次被請求,但返回結果不是不是空列表,而是['a']。如果同一個fixture在同一個測試函數中每次都去請求,那麼必然返回的是空列表。

7、Fixture的後置處理

前面的案例都是加了前置處理,相當於setup(),後置teardown()fixture中是可以通過yield關鍵字和addfinalizer關鍵字來實現

(一)使用yield關鍵字實現後置

  1. 示例
import pytest


@pytest.fixture(autouse=True)
def fixture_3():
    print("這是一個前置處理")
    yield
    print("這是一個後置處理")

def testcase_1(fixture_3):
    print("這是測試用例1")

def testcase_2(fixture_3):
    print("這是測試用例2")

"""
執行結果
fixture/use_fixture_3.py::testcase_1 這是一個前置處理
這是測試用例1
PASSED這是一個後置處理

fixture/use_fixture_3.py::testcase_2 這是一個前置處理
這是測試用例2
PASSED這是一個後置處理
"""
  1. 執行順序
    在前置處理中會根據fixture函數之間的線性關係順序調用的,後置處理順序會反過來
import pytest


@pytest.fixture()
def fixture_1():
    print("這是fixture_1")
    return 1

@pytest.fixture()
def fixture_2(fixture_1):
    print("這是fixture_2的【前置】")
    yield 2
    print("這是fixture_2的【後置】")

@pytest.fixture()
def fixture_add(fixture_1, fixture_2):
    print("這是fixture_add的【前置】")
    yield fixture_1 + fixture_2
    print('這是fixture_add的【後置】')

def test_fixture(fixture_2, fixture_add):
    print("這是測試函數")
    assert fixture_add == 3

"""
執行結果
fixture/yiled_order.py::test_fixture 這是fixture_1
這是fixture_2的【前置】
這是fixture_add的【前置】
這是測試函數
PASSED這是fixture_add的【後置】
這是fixture_2的【後置】
"""

(二)使用addfinalizer關鍵字實現後置

  1. request.addfinalizer把函數變成終結器實現後置處理
import pytest

@pytest.fixture()
def fixture_1(request):
    print("這是fixture_1的【前置】處理")
    def addfinalizer_demo():
        print("這是fixture_1的【後置】處理")

    # 註冊後置處理
    request.addfinalizer(addfinalizer_demo)


def test_fixture(fixture_1):
    print("===這是測試函數===")


"""
執行結果
fixture/use_addfinalizer.py::test_fixture 這是fixture_1的【前置】處理
===這是測試函數===
PASSED這是fixture_1的【後置】處理
"""
  1. request.addfinalizer註冊多個終結器函數
import pytest

@pytest.fixture()
def fixture_1(request):
    print("這是fixture_1的【前置】處理")

    def addfinalizer_demo_1():
        print("這是fixture_1的【後置】處理1")

    def addfinalizer_demo_2():
        print("這是fixture_2的【後置】處理2")

    # 將多個後置處理函數註冊成終結函數
    request.addfinalizer(addfinalizer_demo_1)
    request.addfinalizer(addfinalizer_demo_2)

def test_fixture(fixture_1):
    print("=====這是測試函數=====")


"""
執行結果
fixture/use_more_addfinalizer.py::test_fixture 這是fixture_1的【前置】處理
=====這是測試函數=====
PASSED這是fixture_2的【後置】處理2
這是fixture_1的【後置】處理1
"""
  1. 執行順序
    由上面註冊多個終結器函數示例中可以看到,使用addfinalizer關鍵字處理後置函數的執行順序與註冊順序是反的

(三)yield和addfinalizer的區別

區別在於如果fixture中前置出現異常,yield後置不會執行,而addfinalizer後置會執行。

  • 使用yield
import pytest


@pytest.fixture()
def fixture_yield():
    print("這是fixture_yield前置")
    res = 21/0
    yield
    print("這是fixtue_yield後置")



def test_fixture(fixture_yield):
    print("這是測試函數")

"""
執行結果
yield_error.py::test_fixture ERROR                                       [100%]這是fixture_yield前置

test setup failed
@pytest.fixture()
    def fixture_yield():
        print("這是fixture_yield前置")
>       res = 21/0
E       ZeroDivisionError: division by zero

yield_error.py:12: ZeroDivisionError

"""
  • 使用addfinalizer
import pytest


@pytest.fixture()
def fixture_1(request):
    print("這是fixture_1前置")

    def addfinalizer_demo():
        print("這是fixture_1後置")
    request.addfinalizer(addfinalizer_demo)
    res = 21/0

def test_fixture(fixture_1):
    print("這是測試函數")


"""
執行結果
addfinalizer_error.py::test_fixture ERROR                                [100%]這是fixture_1前置

test setup failed
request = <SubRequest 'fixture_1' for <Function test_fixture>>

    @pytest.fixture()
    def fixture_1(request):
        print("這是fixture_1前置")
    
        def addfinalizer_demo():
            print("這是fixture_1後置")
        request.addfinalizer(addfinalizer_demo)
>       res = 21/0
E       ZeroDivisionError: division by zero

addfinalizer_error.py:16: ZeroDivisionError
這是fixture_1後置
"""

從上面yieldaddfinalizer的示例中可以看出yield前置中出現異常,後置處理沒有被執行,而addfinalizer的前置中出現異常,並沒有影響後置的執行。
註意: 使用addfinalizer一定是後置函數註冊成功後出現異常才不會受影響。

參考:https://zhuanlan.zhihu.com/p/355285675


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 區別一:存儲數據大小不同 1.cookie的存儲數據大小在不能超過4kb,每個頁面最多存儲20個cookie 2.localStorage能達到10mb,sessionStorage能達到5mb,雖然容量比cookie大,但是localStorage是同步執行,太大會影響渲染進度 區別二:相容性 1 ...
  • 本文是深入淺出 ahooks 源碼系列文章的第六篇,該系列已整理成文檔-地址。覺得還不錯,給個 star 支持一下哈,Thanks。 本文已收錄到個人博客中,歡迎關註~ 背景 大家在使用 useEffect 的時候,假如回調函數中使用 async...await... 的時候,會報錯如下。 看報錯, ...
  • 最近開發一款導航的項目需要行駛方向,這裡一般是gps會給我返回航向的,但是公司老系統的資料庫沒有這個數據,就只能自己計算咯 getAngle(lng_a,lat_a, lng_b, lat_b){ var a = (90 - lat_b) * Math.PI / 180; var b = (90 - ...
  • 一.簡介: 本文將完成一個真實業務中的設備上報數據的一個例子,完整的展示後臺服務接收到設備上報的數據後,將數據添加到時序資料庫,並且將數據查詢出來的一個例子。本文所有代碼已經上傳GitHub:https://github.com/Tom-shushu/work-study 下的 iotdb-demo ...
  • 文件的創建: package file; import java.io.File; import java.io.IOException; /* create:創建 new:新 file:文件 使用File新建一個文件 / public class CreateNewFileDemo { publi ...
  • 目錄 一.簡介 二.效果演示 三.源碼下載 四.猜你喜歡 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 基礎 零基礎 OpenGL (ES) 學習路線推薦 : OpenGL (ES) 學習目錄 >> OpenGL ES 轉場 零基礎 O ...
  • Python爬蟲之xpath語法及案例使用 鋼鐵俠的知識庫 2022.08.15 我們在寫Python爬蟲時,經常需要對網頁提取信息,如果用傳統正則表達去寫會增加很多工作量,此時需要一種對數據解析的方法,也就是本章要介紹的Xpath表達式。 Xpath是什麼 XPath,全稱 XML Path La ...
  • Java集合04 9.Set介面方法 Set介面基本介紹 無序(添加和取出的順序不一致),沒有索引 不允許重覆元素,所以最多只有一個null JDK API中介面的實現類有: Set介面的常用方法:和List介面一樣,Set介面也是Collection的子介面,因此,常用方法和Collection接 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...