XUnit测试框架-Python unittest

时间:2022-02-16 21:40:25

选择

语言选择

本次个人作业我选择的语言是Python,了解学习Python有一段时间了但是一直没有练习,所以这次玩蛇,使用的版本是Python3.6。

开发工具选择

我选择的IDE是Pycharm,个人认为Pycharm是一款不错的Py开发工具,用起来得心应手,unittset是py自带的包不需要安装直接引用即可,方便快捷。开发工具截图如下。

XUnit测试框架-Python unittest

Python单元测试框架unittest

unittest简介

TestCase(测试用例)

一个testcase就是一个测试用例,包括测试前环境的搭建setUp,执行测试代码run,测试后环境的还原tearDown,是一个完整的测试单元。

TestSuite(测试套件)

多个testcase的集合

TestLoder

用来加载TestCase到TestSuite中

TextTestRunner

是来执行测试用例的

TextTestResult

保存测试结果的类

TestFixture

测试准备前和执行后要做到的工作

核心工作原理

XUnit测试框架-Python unittest

unittest实例

准备待测方法

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

编写测试方法

test_mathfunc.py

import unittest
from mathfunc import * class TestMathFunc(unittest.TestCase):
#每个测试方法以test开头
def test_add(self):
self.assertEqual(3,add(1,2))
self.assertNotEqual(3,add(2,2)) def test_minus(self):
self.assertEqual(1,minus(3,2)) def test_multi(self):
self.assertEqual(6,multi(2,3)) def test_divide(self):
self.assertEqual(2,divide(6,3))
self.assertEqual(2.5,divide(5,2)) if __name__ == '__main__':
#verbosity 输出详细程度 0 1 2
unittest.main(verbosity=2)

运行结果

测试通过

XUnit测试框架-Python unittest

测试不通过

把除法/改为//(整除)报错如下图

XUnit测试框架-Python unittest

TestSuite

上面的代码运行无序,如果我们写的用例有顺序的话,就需要用TestSuite,被添加到TestSuite中的case会被按照顺序执行。

编写test_suite.py代码如下

 import unittest
from test_mathfunc import TestMathFunc
from HTMLTestRunner import HTMLTestRunner if __name__ == '__main__':
suite = unittest.TestSuite() tests = [TestMathFunc("test_add"), TestMathFunc("test_minus"), TestMathFunc("test_divide")]
#addTests添加多个TestCase
#addTest添加单个TestCase
#suite.addTests(tests)
suite.addTests(unittest.TestLoader().loadTestsFromName('test_mathfunc.TestMathFunc'))
#suite.addTests(unittest.TestLoader().loadTestsFromNames(['test_mathfunc.TestMathFunc')]) 传入列表 runner = unittest.TextTestRunner(verbosity=2)
runner.run(suite)

addTests方法

传入一个TestCase对象的列表。

unittest.TestLoader().loadTestsFromName/Names

传入一个TestCase对象,这个对象里可以包含多个test_开头的方法,每个test_开头的方法处理的时候都可以理解为一个TestCase实例。

TestFixture

在实际测试中,我们可能会遇到这种情况,需要测试的方法中有的需要连接数据库,测试完毕需要还原数据,所以我们就需要一个准备环境的方法(setUp)还有清理环境的方法(TearDown),这就是TestFixture所包含的内容。

修改test_mathfunc.py如下

import unittest
from mathfunc import * class TestMathFunc(unittest.TestCase):
#重写了TestCase的方法
def setUp(self):
print("开始测试之前的环境搭建") def tearDown(self):
print("环境清理") def test_add(self):
self.assertEqual(3,add(1,2))
self.assertNotEqual(3,add(2,2)) def test_minus(self):
self.assertEqual(1,minus(3,2)) def test_multi(self):
self.assertEqual(6,multi(2,3))
#skip装饰器
@unittest.skip("我不想执行除法")
#@unittest.skipIf(condition=,reason=) 当condition为true时跳过
#@unittest.skipUnless(condition=,reason=) 为false时跳过
def test_divide(self):
self.assertEqual(2,divide(6,3))
self.assertEqual(2.5,divide(5,2)) if __name__ == '__main__':
unittest.main(verbosity=2)

在实际测试中我们也可能会遇到这样的情况,开始测试前需要连接数据库,测试结束后关闭连接,不需要还原数据,只在开始和结束各自执行一次即可,setUpClass和tearDownClass的作用就是实现以上功能。

    @classmethod
def setUpClass(cls):
print("开始测试之前的环境搭建统一") @classmethod
def tearDownClass(cls):
print("最后清理环境")

输出结果如下图

XUnit测试框架-Python unittest

没有得到想要的结果,多次输出发现setUp和tearDown输出位置不定,猜想是因为不是单线程执行造成的,所以进行调试结果正确,暂存疑问。

XUnit测试框架-Python unittest

跳过测试

有时候当某些条件的时候我们可能需要跳过测试,unittest也为我们提供了相应的方法

    #@unittest.skipIf(condition=,reason=)   当condition为true时跳过
#@unittest.skipUnless(condition=,reason=) 为false时跳过
def test_divide(self):
self.assertEqual(2,divide(6,3))
self.assertEqual(2.5,divide(5,2))

执行结果如下

XUnit测试框架-Python unittest

把结果保存到文件中

保存到文本文件中

 with open('Unittest.txt','a') as f:
runner = unittest.TextTestRunner(stream=f, verbosity=2)
runner.run(suite)

会在项目目录下生成Unittext.txt文本文件,内容如下

test_add (test_mathfunc.TestMathFunc) ... ok
test_divide (test_mathfunc.TestMathFunc) ... ok
test_minus (test_mathfunc.TestMathFunc) ... ok
test_multi (test_mathfunc.TestMathFunc) ... ok ----------------------------------------------------------------------
Ran 4 tests in 0.000s OK

生成HTML

需要HTMLTsetRunner文件

    with open('HTMLReport.html', 'wb') as f:
runner = HTMLTestRunner(stream=f, title="123", description="test", verbosity=2)
runner.run(suite)

参考自