参数化,就是把测试过程中的数据提取出来,通过参数传递不同的数据来驱动用例运行。其实也就是数据驱动的概念。
在 Unittest 中,我们讲过使用 ddt 库配合 unittest 实现数据驱动。在 Pytest 中并不需要额外的库,通过 pytest.mark.parametrize()即可实现参数化。
方法:
parametrize(argnames, argvalues, indirect=False, ids=None, scope=None)
常用参数:
argnames:参数名
argvalues:参数对应值,类型必须为list或元组
当参数为一个时格式:[value]
当参数个数大于一个时,格式为:[(param_value1,param_value2.....),(param_value1,param_value2.....)]
ids:使用的数据组别名,类型必须为list或元组,与argvalues一一对应,用于使用的数据组的展示
使用方法:
@pytest.mark.parametrize(argnames,argvalues)
️ 参数值为N个,测试方法就会运行N次
在使用pytest.mark.parametrize()传递参数化数据时,测试用例本身必须有参数。
测试用例的参数名必须要跟parametrize传递的参数名保持一致
import pytest
def add(a, b):
return a + b
# 单个参数的情况
@pytest.mark.parametrize('a', (1,2,3,4))
def test_add(a): # => 作为用例参数,接收装饰器传入的数据
print('\na的值:', a)
assert add(a, 1) == a+1
if __name__ == '__main__':
pytest.main(["-s", "demo.py"])
输出结果:
collected 4 items
demo.py
a的值: 1
.
a的值: 2
.
a的值: 3
.
a的值: 4
.
============================== 4 passed in 0.06s ==============================
注意用法,@pytest.mark.parametrize() 装饰器接收两个参数,一个参数是以字符串的形式标识用例函数的参数,第二个参数以列表或元组的形式传递测试数据。
多个参数,@pytest.mark.parametrize()第一个参数依然是字符串, 对应用例的多个参数用逗号分隔。第二个参数为一个二维的列表或者元组
import pytest
# 待测函数
def add(a, b):
return a + b
# 多个参数的情况
@pytest.mark.parametrize('a,b,c', ([1,2,3],[4,5,9],[7,8,15]))
def test_add(a,b,c): # => 作为用例参数,接收装饰器传入的数据
print('\na,b,c的值分别为:', f"{a},{b},{c}")
assert add(a, b) == c
if __name__ == '__main__':
pytest.main(["-s", "demo.py","-v"])
执行后输出:
collecting ... collected 3 items
demo.py::test_add[1-2-3]
a,b,c的值分别为: 1,2,3
PASSED
demo.py::test_add[4-5-9]
a,b,c的值分别为: 4,5,9
PASSED
demo.py::test_add[7-8-15]
a,b,c的值分别为: 7,8,15
PASSED
============================== 3 passed in 0.05s ==============================
以上例子我们是对单个的测试用例进行了数据驱动,如果要对整个测试类操作,其实际上也是对类中的测试方法进行参数化。类中的测试方法的参数必须与@pytest.mark.parametrize()中的标识的参数个数一致且参数名相同。
每个测试用例都会循环执行传入参数的组数的次数
import pytest
def add(a, b):
return a + b
@pytest.mark.parametrize('a, b, c', ([1,2,3],[4,13,9],[7,8,15]))
class TestAbc():
def test_add1(self,a,b,c): # => 作为用例参数,接收装饰器传入的数据
assert add(a, b) == c
def test_add2(self,a,b,c): # => 作为用例参数,接收装饰器传入的数据
assert add(a, c) == b
def test_other(self,a,b,c):
print("\na,b的值分别为:",f"{a},{b}")
if __name__ == '__main__':
pytest.main(["-s", "demo.py","-v"])
通过上面的运行结果,我们可以看到,为了区分参数化的运行结果,在结果中都会显示数据组合而成的名称。
数据短小还好说,如果数据比较长而复杂的话,那么就会很难看。
@pytest.mark.parametrize() 还提供了第三个 ids 参数来自定义显示结果,数据格式为元组或者列表,与传入的第二个参数一一对应,仅作展示所用。
test_add[1-2-3]这是结果中显示的名称,ids 的修改会作用于[]中的显示。
我们下面的示例简单的为数据编了一个号:
import pytest
def add(a, b):
return a + b
# ids 的作用
data = ([1,2,3],[4,13,9],[7,8,15])
ids = [f'data{d}' for d in range(len(data))] # => 生成与数据数量相同的名称列表
# ids = ('data0','data1','data2')
@pytest.mark.parametrize('a, b, c', data, ids=ids)
def test_add1(a,b,c):
assert add(a, b) == c
if __name__ == '__main__':
pytest.main(["-s", "demo.py","-v"])
输出结果:
demo.py::test_add1[data0] PASSED
demo.py::test_add1[data1] FAILED
demo.py::test_add1[data2] PASSED
================================== FAILURES ===================================
在进行自动化测试的时候,一般需要对数据进行一个统筹规划实现以用例数据驱动框架进行测试。
常见的有
excel存放
优点: 便于维护,框架成型之后维护用例的人员没有学习成本。
缺点: 执行效率低,且用例量级大的时候易卡顿。
yaml存放
优点: 便于维护,轻便
缺点: 学习成本,用例编写设置不够直观
json存放
优点: 便于维护,轻便
缺点: 学习成本,用例编写设置不够直观
类属性存放
优点: 调用便利
缺点: 学习成本
上面是我收集的一些视频资源,在这个过程中帮到了我很多。如果你不想再体验一次自学时找不到资料,没人解答问题,坚持几天便放弃的感受的话,可以加入我们扣扣群【313782132 】,里面有各种软件测试资源和技术讨论。
软件测试是IT相关行业中最容易入门的学科~不需要开发人员烧脑的逻辑思维、不需要运维人员24小时的随时待命,需要的是细心认真的态度和IT相关知识点广度的了解,每个测试人员从入行到成为专业大牛的成长路线可划分为:软件测试、自动化测试、测试开发工程师 3个阶段。
如果你不想再体验一次自学时找不到资料,没人解答问题,坚持几天便放弃的感受的话,可以加我们的软件测试交流群,里面有各种软件测试资料和技术交流。