使用ddt实现unittest的参数化测试
# 0. 前言
本文介绍如何使用ddt库来完成unitest的参数化设置。
ddt的github地址 (opens new window)
# 1. 为什么需要参数化
我们在写单测中,需要考虑到各种场景,通过输入各种场景的值执行目的的方法,来判断输出是否是我们所期待的值。
如下代码代码所示,针对large_than_two
方法进行了三种场景的校验写了三个单测,但其中逻辑代码是一致的,而只需要使用不同的参数值进行输入,导致有许多的重复代码进行复制粘贴。
from unittest.case import TestCase
def large_than_two(value) -> bool:
return value > 2
class TestDemoCase(TestCase):
def test_larger_than_two_with_three(self):
self.assertTrue(large_than_two(3))
def test_larger_than_two_with_eight(self):
self.assertTrue(large_than_two(8))
def test_larger_than_two_with_five(self):
self.assertFalse(large_than_two(5))
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
# 2. 使用ddt实现参数化
首先需要通过pip来安装该库
pip install ddt
# 2.1 基本使用
我们在TestCase
上添加ddt
装饰器,然后在单测方法上添加data装饰器,并添加了3种场景的输入参数,如下代码所示:
from unittest.case import TestCase
from ddt import ddt, data
def large_than_two(value) -> bool:
return value > 2
@ddt
class TestDemoCase(TestCase):
@data(3, 8, 5)
def test_larger_than_two(self, value):
self.assertTrue(large_than_two(value))
2
3
4
5
6
7
8
9
10
11
12
13
14
15
执行上面的单测,输入出如下:
============================= test session starts =============================
collecting ... collected 3 items
test_demo.py::TestDemoCase::test_larger_than_two_with_three_1_3 PASSED [ 33%]
test_demo.py::TestDemoCase::test_larger_than_two_with_three_2_8 PASSED [ 66%]
test_demo.py::TestDemoCase::test_larger_than_two_with_three_3_5 PASSED [100%]
============================== 3 passed in 0.02s ==============================
2
3
4
5
6
7
8
会发现,虽然我们写了一个单测用例,但是却执行了3个用例,这是因为ddt会将data的3个参数分别注入到test_larger_than_two
方法中value中并执行单测。
在输出的单测信息中,会输出单测方法+第多少个单测+参数值
来表示当前用例的执行。
通过这种方式可以减少我们的重复代码。
# 2.2 多个值使用参数化
当我们需要在一个单测用例中注入多个值时,可以在data中传入多个元组进行参数化,但执行单例时,会将元组注入到value中,我们将其解开则能拿到多个值。代码如下图所示:
def greater(v1, v2) -> bool:
return v1 > v2
@ddt
class TestDemoCase(TestCase):
@data(
(3, 2),
(4, 1),
(8, 6))
def test_greater(self, value):
v1, v2 = value
self.assertTrue(greater(v1, v2))
2
3
4
5
6
7
8
9
10
11
12
13
14
执行之后输出如下所示:
============================= test session starts =============================
collecting ... collected 3 items
test_demo.py::TestDemoCase::test_greater_1__3__2_ PASSED [ 33%]
test_demo.py::TestDemoCase::test_greater_2__4__1_ PASSED [ 66%]
test_demo.py::TestDemoCase::test_greater_3__8__6_ PASSED [100%]
============================== 3 passed in 0.02s ==============================
2
3
4
5
6
7
8
# 2.3 参数化扁平使用
元组中的数据可以由ddt解开后注入到单测方法中的参数中。只需要在单测方法钱添加unpack
装饰器即可。代码如下所示:
from unittest.case import TestCase
from ddt import ddt, data, unpack
@ddt
class TestDemoCase(TestCase):
@unpack
@data(
(3, 2),
(4, 1),
(8, 6))
def test_greater(self, v1, v2):
self.assertTrue(greater(v1, v2))
2
3
4
5
6
7
8
9
10
11
12
13
14
执行后结果如下:
============================= test session starts =============================
collecting ... collected 3 items
test_demo.py::TestDemoCase::test_greater_1__3__2_ PASSED [ 33%]
test_demo.py::TestDemoCase::test_greater_2__4__1_ PASSED [ 66%]
test_demo.py::TestDemoCase::test_greater_3__8__6_ PASSED [100%]
============================== 3 passed in 0.02s ==============================
2
3
4
5
6
7
8
# 2.4 命名参数
我们还可以给传入的参数进行命名而不是元组的形式,传入的参数名称与单测方法中参数的变量名对应,则不需要对应顺序传入,可读性更强了。代码如下:
@ddt
class TestDemoCase(TestCase):
@unpack
@data(
{"first": 3, "second": 2},
{"first": 4, "second": 1},
{"first": 8, "second": 6}
)
def test_greater(self, second, first):
self.assertTrue(greater(first, second))
2
3
4
5
6
7
8
9
10
11
执行后输出:
============================= test session starts =============================
collecting ... collected 3 items
test_demo.py::TestDemoCase::test_greater_1 PASSED [ 33%]
test_demo.py::TestDemoCase::test_greater_2 PASSED [ 66%]
test_demo.py::TestDemoCase::test_greater_3 PASSED [100%]
============================== 3 passed in 0.01s ==============================
2
3
4
5
6
7
8
# 2.5 从json文件中读取参数进行单测
在某些业务中,输入的参数过于复杂,并且场景繁多,如果将参数数据全部放在单测代码中,则会显得繁重,而且代码不易读,ddt提供了从json文件中读取参数来作为单测的输入数据。
创建data.json文件,其中包含了参数数据。
[
{
"first": 3,
"second": 2
},
{
"first": 4,
"second": 1
},
{
"first": 8,
"second": 6
}
]
2
3
4
5
6
7
8
9
10
11
12
13
14
然后使用file_data作为单测方法的装饰器,并传入上面数据文件的路径。
from ddt import ddt, file_data
@ddt
class TestDemoCase(TestCase):
@file_data("data.json")
def test_greater(self, second, first):
self.assertTrue(greater(first, second))
2
3
4
5
6
7
8
执行成功并输出:
============================= test session starts =============================
collecting ... collected 3 items
test_demo.py::TestDemoCase::test_greater_1 PASSED [ 33%]
test_demo.py::TestDemoCase::test_greater_2 PASSED [ 66%]
test_demo.py::TestDemoCase::test_greater_3 PASSED [100%]
============================== 3 passed in 0.02s ==============================
2
3
4
5
6
7
8
# 3. 总结
本文是介绍ddt的基本并常用的用法,如果想要深入使用可以参考官方文档。
其实ddt有个缺点是不能针对某一个单测方法进行单独的执行,必须要运行整个Unittest class才行,这样在调试的过程中非常不方便。
如果你看到本文其实我比较推荐你使用pytest来替代unittest使用,pytest中也有参数化的使用,并且可以单独的去运行每一个单测。
我是因为在做一个django项目,其中使用的是django test来写单测的,而django test是基于Unittest来实现的,所以只能使用ddt来实现参数化。