XUnit测试框架-Python unittest

选择

语言选择

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

开发工具选择

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

Python单元测试框架unittest

unittest简介

TestCase(测试用例)

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

TestSuite(测试套件)

多个testcase的集合

TestLoder

用来加载TestCase到TestSuite中

TextTestRunner

是来执行测试用例的

TextTestResult

保存测试结果的类

TestFixture

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

核心工作原理

工作原理

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)

运行结果

测试通过

测试通过

测试不通过

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

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("最后清理环境")

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

跳过测试

有时候当某些条件的时候我们可能需要跳过测试,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))

执行结果如下
跳过测试

把结果保存到文件中

保存到文本文件中

 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)

参考自

posted @ 2019-04-03 22:14  林静生寒  阅读(1140)  评论(4编辑  收藏  举报