Pytest框架 — 11、Pytest的标记(二)(parametrize参数化)

1、标记实现参数化

通过@pytest.mark.parametrize(argnames, argvalues, indirect=False, ids=None, scope=None)装饰器来实现参数化。

参数说明:

  • argnames:参数名,使用逗号分隔的列表,或元祖,或字符串,表示一个或多个参数名,【常用】
  • argvalues:参数值,可以是列表、元祖、字典列表、元祖列表,【常用】
  • indirect:中文翻译为中间人,为True时可以对argvalues的参数值进行处理,默认False,【不常用】
    • indirect=True: 且argnames的值为fixture函数名,此时argnames的值变为可执行函数,会将argvalues的参数值当做参数传递给fixture函数进行处理,fixture函数返回处理结果给argnames
    • indirect=False: argnames仅为参数名
  • ids:给用例起别名,字符串列表或数字列表,不设置会自动从测试数据中提取
  • scope:待理解【不常用】

示例1:单个参数

import pytest


phone_list = [
    "13881118888",
    "13012034288",
    "13234324188",
    "13231423288"
]

@pytest.mark.parametrize(argnames="phone_num", argvalues=phone_list)
def test_phone_number(phone_num):  # 注意,这里的参数要和argnames参数名一致
    print(f"正在测试手机号{phone_num}")

"""
执行结果
mark/parametrize/single_param.py::test_phone_number[13881118888] 正在测试手机号13881118888
PASSED
mark/parametrize/single_param.py::test_phone_number[13012034288] 正在测试手机号13012034288
PASSED
mark/parametrize/single_param.py::test_phone_number[13234324188] 正在测试手机号13234324188
PASSED
mark/parametrize/single_param.py::test_phone_number[13231423288] 正在测试手机号13231423288
PASSED
"""

示例2:多个参数

import pytest

user_info = [
    ("张三", "18011111111"),
    ("李四", "18022222222"),
    ("王五", "18033333333")
]

@pytest.mark.parametrize(argnames="name,phonenum", argvalues=user_info)
def test_read_info(name, phonenum):
    print(f"正在读取用户{name},手机号{phonenum}")

"""
执行结果
mark/parametrize/multiple_param.py::test_read_info[\u5f20\u4e09-18011111111] 正在读取用户张三,手机号18011111111
PASSED
mark/parametrize/multiple_param.py::test_read_info[\u674e\u56db-18022222222] 正在读取用户李四,手机号18022222222
PASSED
mark/parametrize/multiple_param.py::test_read_info[\u738b\u4e94-18033333333] 正在读取用户王五,手机号18033333333
PASSED
"""

由上面例子执行结果中可以看到中文参数值乱码了,解决办法如下
方法1. 在pytest.ini中加入disable_test_id_escaping_and_forfeit_all_rights_to_community_support=True
方法2. 在conftest.py中加入

# 收集每一个用例name和nodeid的中文显示,转化为utf-8形式
def pytest_collection_modifyitems(items):
    for item in items:
        item.name = item.name.encode("utf-8").decode("unicode_escape")
        item._nodeid = item.nodeid.encode("utf-8").decode("unicode_escape")

示例3:多个参数化(笛卡尔积)

import pytest


data1 = ['a', 'b', 'c']
data2 = [1, 2]


@pytest.mark.parametrize('test1', data1)
@pytest.mark.parametrize('test2', data2)
def test_param(test1, test2):
    print('\n测试数据:{}-{}'.format(test1, test2))
"""
执行结果
mark/parametrize/multiple_parameterization.py::test_param[1-a] 
测试数据:a-1
PASSED
mark/parametrize/multiple_parameterization.py::test_param[1-b] 
测试数据:b-1
PASSED
mark/parametrize/multiple_parameterization.py::test_param[1-c] 
测试数据:c-1
PASSED
mark/parametrize/multiple_parameterization.py::test_param[2-a] 
测试数据:a-2
PASSED
mark/parametrize/multiple_parameterization.py::test_param[2-b] 
测试数据:b-2
PASSED
mark/parametrize/multiple_parameterization.py::test_param[2-c] 
测试数据:c-2
PASSED
"""

示例4:ids参数给用例起别名

import pytest


user_info = [
    ("张三", "18011111111"),
    ("李四", "18022222222"),
    ("王五", "18033333333")
]

@pytest.mark.parametrize(argnames="name,phonenum", argvalues=user_info, ids=["用户1","用户2","用户3"])
def test_read_info(name, phonenum):
    print(f"正在读取用户{name},手机号{phonenum}")
    
"""
执行结果
mark/parametrize/ids.py::test_read_info[用户1] 正在读取用户张三,手机号18011111111
PASSED
mark/parametrize/ids.py::test_read_info[用户2] 正在读取用户李四,手机号18022222222
PASSED
mark/parametrize/ids.py::test_read_info[用户3] 正在读取用户王五,手机号18033333333
PASSED
"""

示例5:使用indirect处理参数值

import pytest

@pytest.fixture()
def fixture_and_parametrize(request):  # request是关键字不能改变,用来接收参数
    print('邮箱账号为:{}'.format(request.param))
    return request.param + "@qq.com"


@pytest.mark.parametrize('fixture_and_parametrize', ['100203', '466238894', '23942423'],
                         indirect=True)
def test_fixture_and_parametrize_2(fixture_and_parametrize):
    print('拼接后邮箱为:{}'.format(fixture_and_parametrize))

"""
执行结果
mark/parametrize/indirect.py::test_fixture_and_parametrize_2[100203] 邮箱账号为:100203
拼接后邮箱为:100203@qq.com
PASSED
mark/parametrize/indirect.py::test_fixture_and_parametrize_2[466238894] 邮箱账号为:466238894
拼接后邮箱为:466238894@qq.com
PASSED
mark/parametrize/indirect.py::test_fixture_and_parametrize_2[23942423] 邮箱账号为:23942423
拼接后邮箱为:23942423@qq.com
PASSED
"""

示例6:标记数据

在参数化的过程中也可以标记数据进行跳过等

import pytest


@pytest.mark.parametrize("test_input,expected", [
    ("3+9", 12), ("5+4", 9),
    pytest.param("7 * 9", 42, marks=pytest.mark.xfail),
    pytest.param("8 * 6", 42, marks=pytest.mark.skip)
])
def test_mark(test_input, expected):
    assert eval(test_input) == expected
"""
执行结果
mark/parametrize/mark_data.py::test_mark[3+9-12] PASSED
mark/parametrize/mark_data.py::test_mark[5+4-9] PASSED
mark/parametrize/mark_data.py::test_mark[7 * 9-42] XFAIL
mark/parametrize/mark_data.py::test_mark[8 * 6-42] SKIPPED (unconditional skip)
"""

参考
https://www.cnblogs.com/miki-peng/p/14736332.html
https://zhuanlan.zhihu.com/p/515377205

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Pytest框架 — 11、Pytest的标记(二)(parametrize参数化) - Python技术站

(0)
上一篇 2023年4月2日 下午5:10
下一篇 2023年4月2日 下午5:10

相关文章

  • Pytest框架 — 13、Pytest的标记(四)(分组执行)

    目录 1、前言 2、mark的使用 (一)注册自定义标记 (二)在测试用例上标记 (三)执行 3、扩展 (一)在同一个测试用例上使用多个标记 (二)在测试类上使用标记 1、前言 在自动化测试工作中我们有时候并不需要测试所有的测试用例,比如在冒烟测试阶段,我们只需要测试基本功能是否正常就可以了。在pytest中提供了mark标记功能来实现分组执行。 2、mar…

    Python开发 2023年4月2日
    00
  • Pytest框架 — 04、Pytest的断言

    目录 1、前言 2、常用断言 3、预期异常断言 4、优化断言提示 5、自定义失败断言提示 1、前言 Pytest使用python自带关键字assert进行断言,断言内容为一个表达式,只要表达式的最终结果为True,那么断言通过,用例执行成功,否则用例执行失败。assert + 表达式 2、常用断言 assert xx:判断xx为真 assert not xx…

    Python开发 2023年4月2日
    00
  • Pytest框架 — 14、Pytest的标记(五)(控制测试用例执行顺序)

    目录 1、前言 2、使用 3、标记最先执行和最后执行 1、前言 在执行自动化测试时,我们通常都希望能够控制执行测试用例的顺序。 在unittest框架中默认按照ACSII码的顺序加载测试用例并执行,顺序为:0~9、A~Z、a~z,测试目录、测试模块、测试类、测试方法/测试函数都按照这个规则来加载测试用例。 在pytest测试框架中,默认从上至下执行,也可以通…

    Python开发 2023年4月2日
    00
  • Pytest框架 — 09、Pytest的conftest.py文件

    目录 1、conftest.py介绍 2、conftest.py的特点 3、conftest.py文件的使用 4、conftest.py的作用域 1、conftest.py介绍 conftest.py是pytest框架的一种固定写法,把fixture或者自己定义的插件写到这个文件里就会自动去调用。我们前面都是将fixture写到测试用例文件里,在实际工作中更…

    Python开发 2023年4月2日
    00
  • Pytest框架 — 07、Pytest的Fixture(部分前后置)(二)

    目录 4、Fixture的相互调用 5、Fixture复用 6、Fixture缓存返回结果 7、Fixture的后置处理 (一)使用yield关键字实现后置 (二)使用addfinalizer关键字实现后置 (三)yield和addfinalizer的区别 4、Fixture的相互调用 示例: import pytest # 第一层fixture @pyte…

    Python开发 2023年4月2日
    00
  • Pytest框架 — 16、Pytest的测试报告(pytest-html插件和Allure框架)

    目录 1、前言 2、pytest-html生成测试报告 (1)pytest-html插件安装 (2)pytest-html的使用 (3)报告独立显示 3、Allure框架生成测试报告 (1)说明 (2)环境准备 (3)执行测试并生成测试报告 1、前言 在pytest中常用的测试报告生成方法有两种,一种是通过pytest-html插件来完成,另一种通过Allu…

    2023年4月2日
    00
  • Pytest框架 — 05、setup和teardown(全部前后置)

    目录 1、前言 2、各级别详解 3、总结 1、前言 在自动化测试过程中经常需要进行初始化和后期处理等工作,如电商加购物车测试需要先登录,测试完毕需要关闭浏览器等。Pytest提供了5种类型的setup和teardown的方法,具体如下: 模块级别:setup_module,teardown_module 函数级别:setup_function,teardow…

    Python开发 2023年4月2日
    00
  • Pytest框架 — 03、Pytest全局配置文件

    目录 1、Pytest的全局配置文件 (1)说明 (2)示例 (3)常用配置项 (4)执行 1、Pytest的全局配置文件 (1)说明 我们之前用命令行或主函数启动测试的时候会增加一些参数来实现所需功能,很多测试用例执行都需要这些参数,所以在企业中我们会通过全局配置文件的方式来管理这些参数。注意: 全局配置文件的名称是固定的,可以为pytest.ini,to…

    Python开发 2023年4月2日
    00
合作推广
合作推广
分享本页
返回顶部