Python测试框架pytest(17)参数化parametrize

2023-10-27

目录

1、参数 

2、装饰测试类 

3、多个参数化装饰器

4、参数化(传入字典数据) 

5、标记参数化

6、解决unicode编码问题 


@pytest.mark.parametrize 允许在测试函数或类中定义多组参数和 fixtures。

参数化场景:

只有测试数据和预期结果不一样,但操作步骤是一样的测试用例是可以用上参数化的。

创建test_parametrize.py文件

示例一:未参数化

1、脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

# 未参数化
def test_case1():
    assert 3 + 4 == 9

def test_case2():
    assert 2 + 5 == 7

def test_case3():
    assert 6 * 9 == 48

可以看到,三个用例都是先计算,然后断言某个值,重复写三个类似的用例有些冗余。

2、运行结果:

示例二:参数化(优化代码)

1、脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

# 参数化(优化代码)
@pytest.mark.parametrize("test_input, expected", [("3+4", 9), ("2+5", 7), ("6*9", 48)])
def test_eval(test_input, expected):
    print(f"测试数据{test_input}, 预期结果{expected}")
    assert eval(test_input) == expected

2、运行结果:

可以看到,只有一条用例,但是利用参数化输入三组不同的测试数据和预期结果,最终执行的测试用例数还是3条,可以节省很多代码。

1、参数 

def parametrize(self, argnames, argvalues, indirect=False, ids=None, scope=None):

主要参数:

(1)argnames:参数名,是个字符串,如中间用逗号分隔则表示为多个参数名。

1、创建test_parametrize2.py文件

参数名也可以是list或者tuple里的字符串。

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

@pytest.mark.parametrize(["name", "pwd"], [("AllTests", "123456"), ("qq", "85135506")])
def test_case1(name, pwd):
    print(name, pwd)

@pytest.mark.parametrize(("name", "pwd"), [("wangmcn", "567890"), ("admin", "123qwe")])
def test_case2(name, pwd):
    print(name, pwd)

@pytest.mark.parametrize("name, pwd", [("root", "root"), ("guest", "guest")])
def test_case3(name, pwd):
    print(name, pwd)

2、运行结果:

(2)argvalues:参数值,参数组成的列表,列表中有几个元素,就会生成几条用例。

1、创建test_parametrize3.py文件

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

# 如果只有一个参数,里面则是值的列表
@pytest.mark.parametrize("name", ["AllTests", "wangmcn", "admin"])
def test_case1(name):
    print(name)

# 如果有多个参数,list包含tuple
@pytest.mark.parametrize(["name", "pwd"], [("AllTests", "123456"), ("qq", "85135506")])
def test_case2(name, pwd):
    print(name, pwd)

# 如果有多个参数,tuple包含list
@pytest.mark.parametrize(("name", "pwd"), (["wangmcn", "567890"], ["admin", "123qwe"]))
def test_case3(name, pwd):
    print(name, pwd)

# 如果有多个参数,list包含list
@pytest.mark.parametrize("name, pwd", [["root", "root"], ["guest", "guest"]])
def test_case4(name, pwd):
    print(name, pwd)

2、运行结果:

(3)ids:用例的ID。传一个字符串列表,可以标识每一个测试用例,自定义测试数据结果的显示,为了增加可读性。

注:ids的长度需要与测试数据列表的长度一致。

1、创建test_parametrize8.py文件

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

my_data = [(1, 2, 3), (4, 5, 9)]

ids = ["a:{} + b:{} = expect:{}".format(a, b, expect) for a, b, expect in my_data]

@pytest.mark.parametrize("a, b, expect", my_data, ids=ids)
def test_parametrize_case(a, b, expect):
    print("测试数据为{}-{}".format(a, b))
    assert a + b == expect

2、运行结果:

2、装饰测试类 

1、创建test_parametrize2.py文件

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

my_data = [(1, 2, 3), (4, 5, 9)]

@pytest.mark.parametrize("a, b, expect", my_data)
class TestParametrize:

    def test_parametrize_case1(self, a, b, expect):
        print("\n测试用例case1 测试数据为{}+{}".format(a, b))
        assert a + b == expect

    def test_parametrize_case2(self, a, b, expect):
        print("\n测试用例case2 测试数据为{}+{}".format(a, b))
        assert a + b == expect

2、运行结果:

当装饰器 @pytest.mark.parametrize 装饰测试类时,会将数据集合传递给类的所有测试用例方法。

3、多个参数化装饰器

一个函数或一个类可以装饰多个 @pytest.mark.parametrize

当参数化装饰器有很多个的时候,用例数等于n(个)*n(个)*n(个)*n(个)*....

1、创建test_parametrize5.py文件。

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

my_data_1 = [1, 2, 3]
my_data_2 = ["a", "b"]

@pytest.mark.parametrize("a", my_data_1)
@pytest.mark.parametrize("b", my_data_2)
def test_parametrize_case(a, b):
    print(f"测试数据为:{a},{b}")

2、运行结果:

参数a的数据有3个,参数b的数据有2个,所以最终的用例数有3*2=6条。

4、参数化(传入字典数据) 

1、创建test_parametrize6.py文件

数据类型为字典

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

my_data = (
    {
        "user": "AllTests",
        "pwd": "123456"
    },
    {
        "user": "wangmcn",
        "pwd": "567890"
    }
)

@pytest.mark.parametrize('dic', my_data)
def test_parametrize_case(dic):
    print(f'测试数据为\n{dic}')
    print(f'user:{dic["user"]}, pwd:{dic["pwd"]}')

2、运行结果:

5、标记参数化

pytest.param 可以传三种参数:

  • param values - 参数集值的变量 args,按顺序排列。

  • keyword marks - marks 关键字标记,要应用于此参数集的单个标记或标记列表。

  • keyword str id - 参数集的属性 id。

示例一:

1、创建test_parametrize7.py文件。

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

@pytest.mark.parametrize("test_input, expected", [
    ("3+5", 8),
    ("2+4", 6),
    pytest.param("6*9", 45, marks=pytest.mark.xfail),
    pytest.param("6*6", 50, marks=pytest.mark.skip)
])
def test_case(test_input, expected):
    assert eval(test_input) == expected

2、运行结果:

示例二:

1、创建test_parametrize7_2.py文件。

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

@pytest.mark.parametrize("test_input, expected", [
    pytest.param("3+5", 8, id="case1"),
    pytest.param("2+4", 6, id="case2"),
    pytest.param("6*9", 45, marks=pytest.mark.xfail, id="case3"),
    pytest.param("6*6", 50, marks=pytest.mark.skip, id="case4")
])
def test_case(test_input, expected):
    assert eval(test_input) == expected

2、运行结果:

6、解决unicode编码问题 

使用 @pytest.mark.parametrize 参数化的时候,加 ids 参数用例描述有中文时,在控制台输出会显示 unicode 编码问题,中文不能正常显示。

解决方法:使用 pytest_collection_modifyitems 钩子函数,对输出的 item.name 和 item.nodeid 重新编码即可。

1、创建test_parametrize9.py文件

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

import pytest

def login(username, password):
    "登陆操作"
    return {"code": 0, "msg": "success!"}

my_data = [
    ({"username": "AllTests", "password": "123456"}, "success!"),
    ({"username": "Admin123", "password": "123456"}, "success!"),
    ({"username": "Admin456", "password": "123456"}, "success!"),
]

ids = [
    "输入正确a账号,正确密码,登录成功",
    "输入正确b账号,正确密码,登录成功",
    "输入正确c账号,正确密码,登录成功",
]

@pytest.mark.parametrize("input_data, expected", my_data, ids=ids)
def test_login(input_data, expected):
    "测试用例-登陆"
    result = login(input_data["username"], input_data["password"])
    assert result["msg"] == expected

2、打开命令行,输入执行命令

pytest -v test_parametrize9.py

运行结果:

控制台输出中文时显示unicode编码问题。

3、解决ids参数用例描述为中文时,控制台输出显示unicode编码问题。

在项目的根目录或与用例同级目录下,创建conftest.py文件

脚本代码:

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
微信公众号:AllTests软件测试
"""

def pytest_collection_modifyitems(items):
    for item in items:
        item.name = item.name.encode("utf-8").decode("unicode_escape")
        # print(item.nodeid)
        item._nodeid = item.nodeid.encode("utf-8").decode("unicode_escape")

重新执行用例,运行结果:

控制台输出中文时显示正常。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Python测试框架pytest(17)参数化parametrize 的相关文章

随机推荐

  • Java 包之间类引用

    在Java中 包 Package 是一种用于组织和管理代码的机制 一个包中可以包含多个类 而一个类也可以引用属于其他包中的类 本篇博客将介绍Java包之间类引用的基本概念 语法和用法 以及如何在不同的包之间引用类 基本概念 在Java中 一
  • MySQL中如何查看错误日志和二进制日志?

    1 错误日志 错误日志是 MySQL 中最重要的日志之一 它记录了当mysqld启动和停止时 以及服务器在运行过程中发生任何严重错误时的相关信息 当数据库出现任何故障导致无法正常使用时 建议首先查看此日志 该日志是默认开启的 默认存放目录
  • springboot实现多文件上传

    实现springboot的多文件上传 首先创建一个springboot项目 添加spring boot starter web依赖 然后在resources下的static文件夹下创建uploads html文件 文件内容如下
  • element问题总结2

    20 隐藏el calendar 的多余的日子 deep el calendar table not is range td next display none deep el calendar table not is range td
  • 电路分析第二章 网孔电流法,节点电压法,叠加原理

    文章目录 前言 一 网孔分析 二 节点电压法 三 叠加原理 总结 前言 纯电阻电路的分析 就是求解各支路的电压或者电流 具体方法包含 2b法 KCL KVL VCR 其中b是指支路数量 1b法其实就是2b法的化简 先求电压 或者先求电流 若
  • linux - tar压缩解压缩使用,快速记忆方法

    linux常用的指令中 属压缩 解压缩最容易记混 分享一个快速记忆方法 使用最常用 且功能也比较全的 tar命令 1 tar 基本参数 v 显示操作过程 大多数情况都会加上 f 指定压缩文件 必加 j 支持bzip2解压文件 bz bz2后
  • 【毕设选题】flink大数据淘宝用户行为数据实时分析与可视化

    文章目录 0 前言 1 环境准备 1 1 flink 下载相关 jar 包 1 2 生成 kafka 数据 1 3 开发前的三个小 tip 2 flink sql 客户端编写运行 sql 2 1 创建 kafka 数据源表 2 2 指标统计
  • 数据库作业11:SQL练习7 - GRANT/ REVOKE / AUDIT

    授权 授予与收回 GRANT语句向用户授予对数据的操作权限 REVOKE语句向用户收回对数据的操作权限 1 GRANT 语句格式为 GRANT lt 权限 gt lt 权限 gt ON lt 对象类型 gt lt 对象名 gt lt 对象类
  • mediapipe KNN 基于mediapipe和KNN的引体向上计数/深蹲计数/俯卧撑计数【mediapipe】

    原文链接 https blog csdn net m0 57110410 article details 125569971 Mediapipe KNN引体向上计数 深蹲计数 俯卧撑计数 引言 功能 说明 步骤 训练样本 获取归一化的 la
  • 4个数据整理Excel小技巧,省下你80%的工作时间

    各位小伙伴 不知道你觉得工作中哪项任务是最烦的呢 那么今天 将几个特别实用的整理报表数据Excel小技巧带给大家 让我们把复杂的问题简单化 来吧 好好学习 天天向上 更加进步吧 1 输入证件号码 在录入个人身份信息的时候 身份证号有时候是经
  • 又一款 AI 应用开源了,让你的绘画作品动起来!

    这是 进击的Coder 的第 824 篇技术分享 作者 小 G 来源 GitHubDaily 阅读本文大概需要 4 分钟 2021 年的时候 Meta 前身是 Facebook 团队发布了一款非常有趣的 AI 工具 叫 Animated D
  • Excel根据出生日期判断生肖,Leo老师来教你!

    在工作学习中 我们经常会遇到Excel根据出生日期判断生肖这样的问题 列夫托尔斯泰说过 人生不是一种享乐 而是一桩十分沉重的工作 因此 面对Excel根据出生日期判断生肖我们应该有努力探索的精神 成功的人千方百计 失败的人千难万险 对于这个
  • python虚拟环境安装使用

    conda下操作 1 查看已经安装的虚拟环境 conda env list 2 创建 conda create n your env name 3 进入虚拟环境 conda activate your env name 4 退出虚拟环境 c
  • 09.二叉树

    09 二叉树 1 树型结构 1 1概念 树是一种非线性的数据结构 它是由n n gt 0 个有限结点组成一个具有层次关系的集合 把它叫做树是因为它看起来像一棵倒挂的树 也就是说它是根朝上 而叶朝下的 它具有以下的特点 有一个特殊的结点 称为
  • 最小二乘法与伪逆矩阵

    一 简介 最小二乘法是一种数学优化技术 通过最小化误差的平方和寻找数据的最佳函数匹配 利用最小二乘法可以简便地求得未知的数据 并是得这些求得的数据与实际数据之间误差的平方和最小 二 最小二乘法拟合直线的原理 1 假设存在n个坐标点 他们的坐
  • C语言的四种程序结构

    1 顺序结构 顺序结构的程序设计是最简单的 只要按照解决问题的顺序写出相应的语句就行 它的执行顺序是自上而下 依次执行 例如 a 3 b 5 现交换a b的值 这个问题就好像交换两个杯子水 这当然要用到第三个杯子 假如第三个杯子是c 那么正
  • anaconda python未激活_anaconda无法激活新建环境,提示没有那个文件或目录

    遇到的问题 最新在部署tensorflow 但是由于cpu型号比较老的原因 所以直接pip安装tensorflow会提示core dump 吐核 所以需要使用conda来建立一个新的环境 然后使用conda来安装tf即可解决吐核问题 但是有
  • 高效实现延迟消息功能

    高效实现延迟消息功能 高效延时消息 包含两个重要的数据结构 1 环形队列 例如可以创建一个包含3600个slot的环形队列 本质是个数组 2 任务集合 环上每一个slot是一个Set 同时 启动一个timer 这个timer每隔1s 在上述
  • Unity2D入门(七):物理材质、跳跃、基础UI

    一 物理材质 游戏角色在跳跃过程中如果正面碰撞到地形就会卡在上面 所以需要为其添加一种材质 在Assets窗口中 右键 gt 新建一个物理材质 不用做其他的调整 将其拖拽到Player Inspector窗口中的BoxCollider组件的
  • Python测试框架pytest(17)参数化parametrize

    目录 1 参数 2 装饰测试类 3 多个参数化装饰器 4 参数化 传入字典数据 5 标记参数化 6 解决unicode编码问题 pytest mark parametrize 允许在测试函数或类中定义多组参数和 fixtures 参数化场景