unittest中最核心的四个概念是:
test case, test suite, test runner, test fixture。
(1)一个TestCase的实例就是一个测试用例。什么是测试用例呢?就是一个完整的测试流程,包括测试前准备环境的搭建(setUp),执行测试代码(run),以及测试后环境的还原(tearDown)。元测试(unit test)的本质也就在这里,一个测试用例是一个完整的测试单元,通过运行这个测试单元,可以对某一个问题进行验证。
(2)而多个测试用例集合在一起,就是TestSuite,而且TestSuite也可以嵌套TestSuite。
(3)TestLoader是用来加载TestCase到TestSuite中的,其中有几个loadTestsFrom__()方法,就是从各个地方寻找TestCase,创建它们的实例,然后add到TestSuite中,再返回一个TestSuite实例。
(4)TextTestRunner是来执行测试用例的,其中的run(test)会执行TestSuite/TestCase中的run(result)方法。
测试的结果会保存到TextTestResult实例中,包括运行了多少测试用例,成功了多少,失败了多少等信息。
(5)而对一个测试用例环境的搭建和销毁,是一个fixture。
1.基本案例
test_mathfunc.py
# -*- coding:utf-8 -*-
import unittest
from mathfunc import *
'''
mathfunc.py
def add(a, b):
return a+b
def minus(a, b):
return a-b
def multi(a, b):
return a*b
def divide(a, b):
return a/b
'''
# 创建测试类
class TestMathFunc(unittest.TestCase):
""" 用于测试mathfunc.py """
def test_add(self):
# add(a, b) 测试
self.assertEqual(6, add(2, 4)) # 判断6 与 add(2,4)的结果是否一致
self.assertNotEqual(3, add(2, 2)) # 判断3 与add(2,2)结果是否不一致
def test_minus(self):
# minus(a, b)测试
self.assertEqual(3, minus(9, 6)) # 判断3 与 minus(9,6))的结果是否一致
def test_multi(self):
# multi(a, b)测试
self.assertEqual(6, multi(2, 3)) # 判断6 与 minus(2,3))的结果是否一致
def test_divide(self):
# divide(a, b) 测试
self.assertEqual(3, divide(6, 2)) # 判断3 与 minus(6,2))的结果是否一致
self.assertEqual(2.5, divide(5, 2)) # 判断2.5 与 minus(5,2))的结果是否一致
if __name__ == '__main__':
unittest.main() # 执行测试
2.TestSuite案例
按照顺序执行测试用例,并将测试结果输出到指定文件
test_suite.py
# -*- coding:utf-8 -*-
import unittest
from unit_pro import TestMathFunc
if __name__ == '__main__':
suite = unittest.TestSuite() # 创建测试集合
tests = [TestMathFunc("test_add"), TestMathFunc("test_minus"), TestMathFunc("test_divide")] # 测试函数列表
suite.addTests(tests) # 将测试函数列表添加到测试集合中
# 控制台输出测试结果
# runner = unittest.TextTestRunner(verbosity=2)
# verbosity=0 只显示执行的用例的总数和全局的执行结果。
# verbosity=1 默认值,显示执行的用例的总数和全局的执行结果,并对每个用例的执行结果(成功T或失败F)有个标注。
# verbosity=2 显示执行的用例的总数和全局的执行结果,并输出每个用例的详细的执行结果。
# runner.run(suite) # 运行单元测试
# 将测试结果输出到文件
with open('unit_test_report.txt','a') as f:
runner = unittest.TextTestRunner(stream=f, verbosity=2)
runner.run(suite)
3.测试前后的环境准备和清理
setUp用来为测试准备环境,tearDown用来清理环境 注意:每次执行case前后都执行了一次
setUpClass,tearDownClass在整个测试中只执行一次
案例
# -*- coding:utf-8 -*-
import unittest
from mathfunc import *
'''
mathfunc.py
def add(a, b):
return a+b
def minus(a, b):
return a-b
def multi(a, b):
return a*b
def divide(a, b):
return a/b
'''
# 创建测试类
class TestMathFunc(unittest.TestCase):
""" 用于测试mathfunc.py """
# setUp用来为测试准备环境,tearDown用来清理环境 注意:每次执行case前后都执行了一次
# def setUp(self):
# print('执行测试之前,先准备好环境 test.Prepare environment')
#
# def tearDown(self):
# print('执行完成后的清理 test.Clean up')
@classmethod
def setUpClass(cls):
print('测试中,只在开始时被调用一次')
@classmethod
def tearDownClass(cls):
print('测试中,只在结束时被调用一次')
def test_add(self):
# add(a, b) 测试
self.assertEqual(6, add(2, 4)) # 判断6 与 add(2,4)的结果是否一致
self.assertNotEqual(3, add(2, 2)) # 判断3 与add(2,2)结果是否不一致
def test_minus(self):
# minus(a, b)测试
self.assertEqual(3, minus(9, 6)) # 判断3 与 minus(9,6))的结果是否一致
def test_multi(self):
# multi(a, b)测试
self.assertEqual(6, multi(2, 3)) # 判断6 与 minus(2,3))的结果是否一致
def test_divide(self):
# divide(a, b) 测试
self.assertEqual(3, divide(6, 2)) # 判断3 与 minus(6,2))的结果是否一致
self.assertEqual(2.5, divide(5, 2)) # 判断2.5 与 minus(5,2))的结果是否一致
if __name__ == '__main__':
unittest.main() # 执行测试
4.跳过某个case(实例)
skip无条件跳过,skipIf当condition为True时跳过,skipUnless当condition为False时跳过。
@unittest.skip("我要跳过这个测试实例") # 跳过该实例
def test_divide(self):
# divide(a, b) 测试
self.assertEqual(3, divide(6, 2)) # 判断3 与 minus(6,2))的结果是否一致
self.assertEqual(2.5, divide(5, 2)) # 判断2.5 与 minus(5,2))的结果是否一致