Unittest是python内置的一个单元测试框架,主要用于自动化测试用例的开发与执行
简单的使用如下
import unittest
class TestStringMethods(unittest.TestCase):
def setUp(self):
print("test start")
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
def tearDown(self):
print("test end")
if __name__ == '__main__':
unittest.main()
运行测试
1.使用命令行python -m unittest xxx脚本名
2.有unittest.main()就直接执行脚本
结果
----------------------------------------------------------------------
test start
test end
.test start
test end
.
----------------------------------------------------------------------
Ran 2 tests in 0.001s
OK
整体结构:unittest库提供了Test Case, Test Suite, Test Runner, Test Fixture
构建用例的方法主要就是继承TestCase类,创建自己的测试类,然后用约定的test开头命名方法,这些方法就是测试用例
class TestStringMethods(unittest.TestCase):
#用例以test开头
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
写用例的时候常使用断言,主要断言有以下:
断言方法 | 检查条件 |
assertEqual(a, b) | a == b |
assertNotEqual(a, b) | a != b |
assertTrue(x) | bool(x) is True |
assertFalse(x) | bool(x) is False |
assertIs(a, b) | a is b |
assertIsNot(a, b) | a is not b |
assertIsNone(x) | x is None |
assertIsNotNone(x) | x is not None |
assertIn(a, b) | a in b |
assertNotIn(a, b) | a not in b |
assertlsInstance(a, b) | isinstance(a, b) |
assertNotIsInstance(a, b) | not isinstance(a, b) |
还有判断数据类型的断言:
断言方法 | 用于比较的类型 |
assertMultiLineEqual(a, b) | 字符串(string) |
assertSequenceEqual(a, b) | 序列(sequence) |
assertListEqual(a, b) | 列表(list) |
assertTupleEqual(a, b) | 元组(tuple) |
assertSetEqual(a, b) | 集合(set 或 frozenset) |
assertDictEqual(a, b) | 字典(dict) |
官网还给了剩下其他的断言,比如异常,日志等,可以查看unittest --- 单元测试框架 — Python 3.11.0 文档
1.用例环境清理
每个用例执行的时候需要独特的测试环境,可以在单独test方法中编写,但是每个用例执行前后的环境清理或统一的预处理,需要特殊的Test Fixture方法解决
主要使用这两种方法
class TestStringMethods(unittest.TestCase):
def setUp(self):
print("test start")
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
def tearDown(self):
print("test end")
2.用例类的环境清理
上面说的是每个测试用例(以 test_ 开头的方法)的环境清理,那么每个测试类(继承TestCase 的类)运行的时候怎么清理环境呢?
主要使用下面两个类方法
class TestStringMethods(unittest.TestCase):
@classmethod
def setUpClass(self):
print("test start")
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
@classmethod
def tearDownClass(self):
print("test end")
3.模块级的环境清理
运行多个测试的时候,可能会将一部分功能的测试类集中在一个文件中,对这一个文件级的环境清理主要使用下面两种方法
./test.py
def setUpModule():
print('test module start')
def tearDownModule():
print("test module end")
class Test1(unittest.TestCase):
...
class Test2(unittest.TestCase):
...
1.通过代码调用测试用例
if __name__ == '__main__':
unittest.main()
2.命令行执行
#运行测试文件
python -m unittest test_module
#测试单个测试类
python -m unittest test_module.test_class
#测试多个测试类
python -m unittest test_module.test_class test_module2.test_class2
#通配符匹配测试文件执行
python -m unittest -p test*.py
#显示详细信息
python -m unittest -v test_module
#帮助
python -m unittest -h
3.通过组织Test Suit后使用Test Runner运行Suite来运行测试
组织Suite的方法很多,下面怎么组织Suite在管理用例中会介绍
class TestStringMethods(unittest.TestCase):
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
if __name__ == '__main__':
#构建测试集
suite = unittest.TestSuite()
suite.addTest(TestStringMethods("test_upper"))
#执行测试
runner = unittest.TextTestRunner()
runner.run(suite)
通过组织TestSuite可以管理多个测试用例的执行,然后使用Test Runner运行Suite来运行测试,主要用的对象有:
1.通过addTest()的方式
if __name__ == '__main__':
#构建测试集
suite = unittest.TestSuite()
suite.addTest(TestStringMethods("test_upper"))
#执行测试
runner = unittest.TextTestRunner()
runner.run(suite)
2.通过TestLoader()方式组织TestSuite
class TestStringMethods(unittest.TestCase):
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
class TestStringMethods2(unittest.TestCase):
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
if __name__ == '__main__':
#此用法可以同时测试多个类
suite1 = unittest.TestLoader().loadTestsFromTestCase(TestStringMethods)
suite2 = unittest.TestLoader().loadTestsFromTestCase(TestStringMethods2)
suite = unittest.TestSuite([suite1, suite2])
unittest.TextTestRunner().run(suite)
3.统一管理测试用例执行,比如测试用例达到成百上千个,可以将这些用例按照所测试的功能进行拆分,分散到不同的测试文件中,最后再创建用于执行所有测试用例的runtest.py文件
比如有下面很多测试文件
├─test
│ test1.py
│ test2.py
│ test3.py
│ test4.py
│ tmp1.py
│ tmp2.py
...
如果我们只想执行test开头的测试文件,除了上文中的命令行命令外,我们还可以使用defaultTestLoader类提供的discover()方法来加载所有的测试用例
discover(start_dir,pattern='test*.py',top_level_dir=None)
找到指定目录下所有测试模块,并可递归查到子目录下的测试模块,只有匹配到文件名才能被加载。如果启动的不是顶层目录,那么顶层目录必须单独指定。
注意:discover()方法中的start_dir只能加载当前目录下的.py文件,如果加载子目录下的.py文件,需在每个子目录下放一个_init_.py文件。
-runtest.py
import unittest
test_dir = './'
discover = unittest.defaultTestLoader.discover(test_dir, pattern='test*.py')
if __name__ == '__main__':
runner = unittest.TextTestRunner()
runner.run(discover)
当然,如果用例少,也可以使用addTest()的方式一个个添加到TestSuite中
4.跳过用例和预期失败
unittest提供了实现某些需求的装饰器,在执行测试用例时每个装饰前面加@符号。
class TestStringMethods(unittest.TestCase):
@unittest.skip("not wht")
def test_upper(self):
self.assertEqual('foo'.upper(), 'FOO')
5.执行顺序
unittest框架默认根据ASCII码的顺序加载测试用例,数字与字母的顺序为:0-9,A-Z,a-z。所以上文测试方法test_isupper()会比test_upper()先执行
同理测试类以及测试文件也是按照这个顺序执行,但如果你使用addTest()的方式添加了测试,会按照添加的顺序执行
1.console输出结果
结果中有几个特殊字符表示不同的意思
2.HTMLTestRunner输出测试报告
HTMLTestRunner是一个第三方库用于替代TestRunner,用于生成可视化的报表,是python2时期的产物,现在python3需要修改其内容才能用,不过网上有改好的,可以直接用
使用就是将下载好的HTMLTestRunner.py复制到...\python35\Lib目录下,然后下面这样使用
class TestStringMethods(unittest.TestCase):
def test_upper(self):
#HTMLTestRunner可以读取docstring类型的注释
'''
test1
'''
self.assertEqual('foo'.upper(), 'FOO')
def test_isupper(self):
self.assertTrue('FOO'.isupper())
self.assertFalse('Foo'.isupper())
if __name__ == '__main__':
testsuite = unittest.TestSuite()
testsuite.addTest(TestStringMethods("test_upper"))
testsuite.addTest(TestStringMethods("test_isupper"))
fp = open('./result.html', 'wb')
runner = HTMLTestRunner(stream=fp, title='测试报告', description='测试执行情况')
runner.run(testsuite)
fp.close()
?现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
分享他们的经验,还会分享很多直播讲座和技术沙龙
可以免费学习!划重点!开源的!!!
qq群号:485187702【暗号:csdn11】
最后感谢每一个认真阅读我文章的人,看着粉丝一路的上涨和关注,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走!?希望能帮助到你!【100%无套路免费领取】