单元测试框架是一种软件测试方法,通过来测试源代码中的各个单元,例如类,方法等,以确定它们是否符合要求。直观上来说,可以将单元视为最小的可测试部分。单元测试是程序员在开发过程中创建的短代码片段。 它构成了组件测试的基础。
unitest是python内置的用于测试代码的模块,它支持测试自动化,配置共享和关机代码测试。支持将测试样例聚合到测试集中,并将测试与报告框架独立。
为了实现这些,unittest 通过面向对象的方式支持了一些重要的概念。
步骤:
代码:(文件名为unittest01.py)
import unittest
def add(a, b):
return a+b
# 一个类是一个测试用例,测试指类中的测试方法,测试用力中可以有多个测试。
class SimpleTest(unittest.TestCase):
def test_add(self):
self.assertEqual(add(3, 7), 10)
def test_add2(self):
self.assertEqual(add(3, 5), 10)
# __name__是一个全局变量,它的值是模块的名称。该判断语句表示只运行当前文件中的代码。
if __name__ == "__main__":
unittest.main()
结果:
.F
======================================================================
FAIL: test_add2 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/hookind/Desktop/workspace/pythonTest/unittest01.py", line 13, in test_add2
self.assertEqual(add(3, 5), 10)
AssertionError: 8 != 10
----------------------------------------------------------------------
Ran 2 tests in 0.001s
FAILED (failures=1)
可以在命令行使用unittest模块运行模块、类和独立测试方法中的测试(这里测试指指测试用例类中的测试方法):
python -m unittest test1
python -m unittest test_module.TestClass
python -m unittest test_module.TestClass.test_method
示例:对于上面的代码,可以通过下面这些命令运行。
python -m unittest unittest01
python3 -m unittest unittest01.SimpleTest
python3 -m unittest unittest01.SimpleTest.test_add
unittest支持一些命令行选项,可以通过下面的命令查看所有选项。
python3 -m unittest -h
这章只描述unittest模块中的类与方法。
该类的对象表示最小的可测试单元。
在TestCase类中的一些方法。
编号
方法
描述
1
setUp()
用来准备测试脚手架的方法,在调用测试方法之前,该方法会被立即调用
2
tearDown()
测试方法被调用后,该方法会被立即调用,并且结果也会被记录。即使测试方法抛出异常,该方法也会被调用
3
setUpClass()
一个类方法,在类中所有的测试运行前被调用
4
tearDownClass()
一个类方法,在类中所有的测试运行后被调用
脚手架(Fixture)
在测试用例类中,可以有大量的测试。这些方法可能需要初始化数据库连接、临时文件或其它的资源。这些就是脚手架。测试用力需要一些特别的方法配置和清理测试需要的脚手架。为了配置脚手架,覆盖setUp()方法;为了清理,覆盖tearDown()方法。
类脚手架(Class Fixture)
TestCase类有一个setUpClass()方法,该方法可以在TestCase类中的测试执行之前被覆盖以执行。同样,tearDownClass() 方法将在类中的所有测试之后执行。这两种方法都是类方法。因此,它们必须@classmethod注解装饰。
示例:
import unittest
def add(a, b):
return a+b
class SimpleTest(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
print("==========setUpClass=========")
return super().setUpClass()
@classmethod
def tearDownClass(cls) -> None:
print("\n==========tearDownClass=========")
return super().tearDownClass()
def setUp(self):
print("\n==========setUp=========")
return super().setUp()
def tearDown(self):
print("==========tearDown========")
return super().tearDown()
def test_add(self):
print("\n执行test_add")
self.assertEqual(add(3, 7), 10)
def test_add2(self):
print("\n执行test_add2")
self.assertEqual(add(3, 5), 10)
# __name__是一个全局变量,它的值是模块的名称。该判断语句表示只运行当前文件中的代码。
if __name__ == "__main__":
unittest.main()
结果:
==========setUpClass=========
==========setUp=========
执行test_add
==========tearDown========
.
==========setUp=========
执行test_add2
==========tearDown========
F
==========tearDownClass=========
======================================================================
FAIL: test_add2 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/hookind/Desktop/workspace/pythonTest/unittest01.py", line 34, in test_add2
self.assertEqual(add(3, 5), 10)
AssertionError: 8 != 10
----------------------------------------------------------------------
Ran 2 tests in 0.001s
FAILED (failures=1)
想了解更多,请参考unittest.TestCase
Python 的测试框架提供了一种有用的机制,通过该机制可以根据它们测试的特性将测试用例实例组合在一起。该机制由unittest模块中的TestSuite类提供。
步骤:
创建一个TestSuite类的实例。
suite = unittest.TestSuite()
把测试(测试用例中的一个测试方法)加入TestSuite实例中。
suite.addTest(SimpleTest2("test_sub"))
suite.addTest(unittest.makeSuite(SimpleTest))
创建一个TextTestRunner类的对象。
runner = unittest.TextTestRunner()
调用run()方法去运行在套件中的测试。
runner.run(suite)
示例:
import unittest
def add(a, b):
return a+b
def sub(a, b):
return a-b
class SimpleTest(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
print("==========setUpClass=========")
return super().setUpClass()
@classmethod
def tearDownClass(cls) -> None:
print("\n==========tearDownClass=========")
return super().tearDownClass()
def setUp(self):
print("\n==========setUp=========")
return super().setUp()
def tearDown(self):
print("==========tearDown========")
return super().tearDown()
def test_add(self):
print("\n执行test_add")
self.assertEqual(add(3, 7), 10)
def test_add2(self):
print("\n执行test_add2")
self.assertEqual(add(3, 5), 10)
class SimpleTest2(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
print("==========setUpClass=========")
return super().setUpClass()
@classmethod
def tearDownClass(cls) -> None:
print("\n==========tearDownClass=========")
return super().tearDownClass()
def setUp(self):
print("\n==========setUp=========")
return super().setUp()
def tearDown(self):
print("==========tearDown========")
return super().tearDown()
def test_sub(self):
print("\n执行test_sub")
self.assertEqual(sub(3, 5), 10)
def test_sub2(self):
print("\n执行test_sub2")
self.assertEqual(sub(3, 5), -2)
# __name__是一个全局变量,它的值是模块的名称。该判断语句表示只运行当前文件中的代码。
if __name__ == "__main__":
suite = unittest.TestSuite()
suite.addTest(SimpleTest2("test_sub"))
suite.addTest(unittest.makeSuite(SimpleTest))
runner = unittest.TextTestRunner()
runner.run(suite)
结果:
==========setUpClass=========
==========setUp=========
执行test_sub
==========tearDown========
F
==========tearDownClass=========
==========setUpClass=========
==========setUp=========
执行test_add
==========tearDown========
.
==========setUp=========
执行test_add2
==========tearDown========
F
==========tearDownClass=========
======================================================================
FAIL: test_sub (__main__.SimpleTest2)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/hookind/Desktop/workspace/pythonTest/unittest01.py", line 61, in test_sub
self.assertEqual(sub(3, 5), 10)
AssertionError: -2 != 10
======================================================================
FAIL: test_add2 (__main__.SimpleTest)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/hookind/Desktop/workspace/pythonTest/unittest01.py", line 38, in test_add2
self.assertEqual(add(3, 5), 10)
AssertionError: 8 != 10
----------------------------------------------------------------------
Ran 3 tests in 0.001s
FAILED (failures=2)
想了解更多,可参考unittest.TestSuite
unittest包具有TestLoader类,用于从类和模块创建测试套件。默认情况下,在调用unittest.main()方法时会自动创建unittest.defaultTestLoader实例。但是,显式实例允许自定义某些属性。
示例:
import unittest
def add(a, b):
return a+b
def sub(a, b):
return a-b
class SimpleTest(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
print("==========setUpClass=========")
return super().setUpClass()
@classmethod
def tearDownClass(cls) -> None:
print("\n==========tearDownClass=========")
return super().tearDownClass()
def setUp(self):
print("\n==========setUp=========")
return super().setUp()
def tearDown(self):
print("==========tearDown========")
return super().tearDown()
def test_add(self):
print("\n执行test_add")
self.assertEqual(add(3, 7), 10)
def test_add2(self):
print("\n执行test_add2")
self.assertEqual(add(3, 5), 10)
class SimpleTest2(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
print("==========setUpClass=========")
return super().setUpClass()
@classmethod
def tearDownClass(cls) -> None:
print("\n==========tearDownClass=========")
return super().tearDownClass()
def setUp(self):
print("\n==========setUp=========")
return super().setUp()
def tearDown(self):
print("==========tearDown========")
return super().tearDown()
def test_sub(self):
print("\n执行test_sub")
self.assertEqual(sub(3, 5), 10)
def test_sub2(self):
print("\n执行test_sub2")
self.assertEqual(sub(3, 5), -2)
# __name__是一个全局变量,它的值是模块的名称。该判断语句表示只运行当前文件中的代码。
if __name__ == "__main__":
testCaseList = [SimpleTest, SimpleTest2]
testLoad = unittest.TestLoader()
testSuiteList=[]
for testCase in testCaseList:
testSuite = testLoad.loadTestsFromTestCase(testCase)
testSuiteList.append(testSuite)
newSuite = unittest.TestSuite(testSuiteList)
runner = unittest.TextTestRunner()
runner.run(newSuite)
想了解更多,可参考unittest.TestLoader
此类用于编译有关已成功测试和已失败测试的信息。TestResult对象存储一组测试的结果。TextTestRunner.run() 方法返回一个 TestResult 实例。
可以在上面代码中使用下述代码获得TestResult对象。
result = runner.run(newSuite)
更多请参考unittest.TestResult
Python测试框架使用Python的内置assert()函数来测试特定条件。如果断言失败,将引发AssertionError。然后测试框架会将测试标识为失败。其他异常被视为错误。
unittest模块中定义了以下三种断言函数:
基本断言函数评估操作的结果是 True 还是 False。所有断言方法都接受一个msg参数,如果指定,则用作失败时的错误消息。
示例:
import unittest
class TestCase01(unittest.TestCase):
def test_equal(self):
self.assertEqual(12,13,"不相等")
def test_equal2(self):
self.assertEqual(12,13)
def test_equal3(self):
self.assertEqual(12,12)
if __name__=="__main__":
unittest.main()
** 结果**:
FF.
======================================================================
FAIL: test_equal (__main__.TestCase01)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/hookind/Desktop/workspace/pythonTest/unittest03.py", line 6, in test_equal
self.assertEqual(12,13,"不相等")
AssertionError: 12 != 13 : 不相等
======================================================================
FAIL: test_equal2 (__main__.TestCase01)
----------------------------------------------------------------------
Traceback (most recent call last):
File "/home/hookind/Desktop/workspace/pythonTest/unittest03.py", line 8, in test_equal2
self.assertEqual(12,13)
AssertionError: 12 != 13
----------------------------------------------------------------------
Ran 3 tests in 0.000s
FAILED (failures=2)
想了解更多,请参考assert-methods
手机扫一扫
移动阅读更方便
你可能感兴趣的文章