zoukankan      html  css  js  c++  java
  • unittest单元测试框架

    单元测试是指对软件中的最小可测试单元进行检查和验证。不同编程语言有不同的单元测试框架,Java有Junit,TestNg。python中有unittest,Pyunit,testtools。
    单元测试框架的作用:提供用例组织与执行;提供丰富的断言方法;提供丰富的日志与测试结果。

    一、unittest核心要素
    1、TestCase
    一个TestCase的实例就是一个测试用例。测试用例就是一个完整的测试流程,包括测试前准备环境的准备(setUp),执行测试代码(run),以及测试后环境的还原(teardDown)。单元测试(unit test)的本质也就在这里,一个测试用例是一个完整的测试单元,通过运行这个测试单元,可以对某一个问题进行验证。
    2、TestSuite
    多个测试用例集合在一起,就是TestSuite,而且TestSuite也可以嵌套TestSuite。TestLoader是用来加载测试用例到TestSuite中的。
    3、TextTestRunner
    TextTestRunner是来执行测试用例的,其中的run()会执行TestSuite或TestCase中的run(result)方法。测试的结果会保存到TextTestResult实例中,包括运行了多少测试用例,成功了多少,失败了多少等信息。
    4、Fixture
    对一个测试用例环境的搭建和销毁是一个fixture。

    calendar.py

    class Math(object):
        def __init__(self, a, b):
            self.a = int(a)
            self.b = int(b)
    
        def add(self):
            return self.a + self.b
    

     test_Math.py

    from calendar import Math
    import unittest
    
    
    class TestMath(unittest.TestCase):
        def setUp(self):
            print("test start")
    
        def test_add(self):
            j = Math(5, 10)
            self.assertEqual(j.add(), 15)
            # 用例失败场景
            # self.assertEqual(j.add(),12)
    
        def tearDown(self):
            print("test end")
    
    
    if __name__ == '__main__':
        # 构造测试集
        suite = unittest.TestSuite()
        suite.addTest(TestMath("test_add"))
    
        # 执行测试
        runner = unittest.TextTestRunner()
        runner.run(suite)
    

     二、断言

    断言内容是自动化脚本的重要内容,正确设置断言以后才能帮助我们判断测试用例执行结果。
    断言方法:
    assertEqual(a,b) 判断 a==b
    assertNotEqual(a,b) 判断 a!=b
    assertTrue(x) bool(x) is True
    assertFalse(x) bool(y) 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
    assertIsInstance(a,b) isinstance(a,b)
    assertNotIsInstance(a,b) not isinstance(a,b)

    from calendar import Math
    import unittest
    
    
    class TestMath(unittest.TestCase):
        def setUp(self):
            print("test start")
    
        def test_add(self):
            j = Math(5, 10)
            self.assertEqual(j.add(), 15)
            # 用例失败场景
            # self.assertEqual(j.add(),12)
    
        def test_add_not_equal(self):
            j = Math(5, 10)
            self.assertNotEqual(j.add(), 12)
    
        def test_add_true(self):
            j=Math(5,10)
            self.assertTrue(j.add()>10)
    
        def test_in(self):
            self.assertIn("zxw","wo yao zxw")
    
        def test_is(self):
            self.assertIs("zxw","zxwa")
    
        def tearDown(self):
            print("test end")
    
    
    if __name__ == '__main__':
        # 构造测试集
        suite = unittest.TestSuite()
        suite.addTest(TestMath("test_add"))
        suite.addTest(TestMath("test_add_not_equal"))
        suite.addTest(TestMath("test_add_true"))
        suite.addTest(TestMath("test_in"))
        suite.addTest(TestMath("test_is"))
    
        # 执行测试
        runner = unittest.TextTestRunner()
        runner.run(suite)
    

    三、新增用例管理

    前面是针对单个add方法来进行单元测试,如果需要多个方法来进行测试,该如何处理?如新增一个sub方法来进行单元测试验证。

    import unittest
    from calendar import *
    
    
    class Test_add(unittest.TestCase):
        def setUp(self):
            print("test is start")
    
        def test_add(self):
            j = Math(5, 5)
            self.assertEqual(j.add(), 10)
    
        def test_add1(self):
            j = Math(10, 20)
            self.assertEqual(j.add(), 30)
    
        def tearDown(self):
            print("test is end")
    
    
    class Test_sub(unittest.TestCase):
        def setUp(self):
            print("sub test is start")
    
        def test_sub(self):
            i = Math(8, 8)
            self.assertEqual(i.sub(), 0)
    
        def test_sub1(self):
            i = Math(5, 3)
            self.assertEqual(i.sub(), 2)
    
        def tearDown(self):
            print("sub test is end")
    
    
    if __name__ == '__main__':
        suite = unittest.TestSuite
        suite.addTest(Test_add("test_add"))
        suite.addTest(Test_add("test_add1"))
        suite.addTest(Test_sub("test_sub"))
        suite.addTest(Test_sub("test_sub1"))
    
        runner = unittest.TextTestRunner()
        runner.run(suite)
    

    四、用例公共部分hebing

    前面每个测试类都有setUp()和tearDown()方法,而且两个方法内容都是一样的,用于打印开始与结束提示语句,是否可以合并在一起呢?

    from calendar import *
    import unittest
    
    
    class Test_StartEnd(unittest.TestCase):
        def setUp(self):
            print('test start')
    
        def tearDown(self):
            print("test end")
    
    
    class Test_add(Test_StartEnd):
        def test_add(self):
            j = Math(5, 5)
            self.assertEqual(j.add(), 10)
    
    
    class Test_sub(Test_StartEnd):
        def test_sub(self):
            i = Math(8, 8)
            self.assertEqual(i.sub(), 0)
    
    
    if __name__ == '__main__':
        unittest.main()
    

     五、用例执行顺序

    观察如下测试脚本,思考测试用例执行顺序。

    import unittest
    
    
    class Test1(unittest.TestCase):
        def setUp(self):
            print("Test1 start")
    
        def test_c(self):
            print("test_c")
    
        def test_b(self):
            print("test_b")
    
        def tearDown(self):
            print("Test2 end")
    
    
    class Test2(unittest.TestCase):
        def setUp(self):
            print("Test2 start")
    
        def test_d(self):
            print("test_d")
    
        def test_a(self):
            print("test_a")
    
        def tearDown(self):
            print("Test2 end!")
    
    
    if __name__ == '__main__':
        unittest.main()
    

     执行顺序规则——测试类或测试方法的数字与字母顺序0~9,A-Z。

    自定义用例执行顺序:按照unittest.TestSuite的addTest()方法添加的顺序执行。

    import unittest
    
    
    class Test1(unittest.TestCase):
        def setUp(self):
            print("Test1 start")
    
        def test_c(self):
            print("test_c")
    
        def test_b(self):
            print("test_b")
    
        def tearDown(self):
            print("Test2 end")
    
    
    class Test2(unittest.TestCase):
        def setUp(self):
            print("Test2 start")
    
        def test_d(self):
            print("test_d")
    
        def test_a(self):
            print("test_a")
    
        def tearDown(self):
            print("Test2 end!")
    
    
    if __name__ == '__main__':
        suite = unittest.TestSuite
        suite.addTest(Test2("test_d"))
        suite.addTest(Test1("test_c"))
        suite.addTest(Test2("test_a"))
        suite.addTest(Test1("test_b"))
    
        runner = unittest.TextTestRunner()
        runner.run(suite)
    

    六、用例综合框架管理

    前面测试用例与执行都是写在一个文件,当用例数量不断增加的时候,用例的执行与管理变得非常麻烦,因此需要对用例根据具体的功能模块来使用单独的模块来管理。
    案例:Test_Project文件目录下包含4个python文件:
    calculator.py   ——加减法运算方法的实现
    start_end.py   ——SetUp与TearDown管理
    test_add.py    ——加法测试用例
    test_sub.py    ——减法测试用例
    runtest.py      ——用例执行管理

    calculator.py 

    class Math(object):
        def __init__(self, a, b):
            self.a = int(a)
            self.b = int(b)
    
        def add(self):
            return self.a + self.b
    
        def sub(self):
            return self.a - self.b 

    start_end.py 

    import unittest
    
    
    class Setup_tearDown(unittest.TestCase):
        def setUp(self):
            print("start test")
    
        def tearDown(self):
            print("end test")

    test_add.py 

    from calculator import *
    from start_end import *
    
    
    class Test_add(Setup_tearDown):
        def test_add(self):
            j = Math(5, 5)
            self.assertEqual(j.add(), 10)
    
        def test_add1(self):
            j = Math(8, 8)
            self.assertEqual(j.add(), 16)
    

     test_sub.py 

    from calculator import *
    from start_end import *
    
    
    class Test_sub(Setup_tearDown):
        def test_sub(self):
            i = Math(5, 3)
            self.assertEqual(i.sub(), 2)
    
        def test_add1(self):
            i = Math(8, 8)
            self.assertEqual(i.sub(), 0)
    

    runtest.py 

    '''
    使用discover可以一次调用多个脚本
    test_dir被测试脚本的路径
    pattern脚本名称匹配规则
    '''
    
    import unittest
    
    test_dir = './'
    
    discovery = unittest.defaultTestLoader.discover(test_dir, pattern="test*.py")
    
    if __name__ == '__main__':
        runner = unittest.TextTestRunner()
        runner.run(discovery)
    

     七、跳过测试和预期失败

    unittest.skip() 直接跳过测试
    unittest.skipIf() 条件为真,跳过测试
    unittest.skipUnless() 条件为假,跳过测试
    unittest.expectedFailure 预期设置失败

    import unittest
    
    
    class Test1(unittest.TestCase):
        # 类运行之前运行次方法
        @classmethod
        def setUpClass(cls):
            print("class module start test>>>>>>>")
    
        # 类中的所有方法执行完成之后执行此方法
        @classmethod
        def tearDownClass(cls):
            print("class module end test<<<<<<<<")
    
        def setUp(self):
            print("Test1 start")
    
        @unittest.skipIf(4 > 3, "skip test_c")
        def test_c(self):
            print("test_c")
    
        @unittest.skipUnless(0 > 1, "skip test_b")
        def test_b(self):
            print("test_b")
    
        def tearDown(self):
            print("Test2 end")
    
    
    # @unittest.skip("skip test2")
    class Test2(unittest.TestCase):
        def setUp(self):
            print("Test2 start")
    
        def test_d(self):
            print("test_d")
    
        @unittest.expectedFailure
        def test_a(self):
            print("test_a")
    
        def tearDown(self):
            print("Test2 end!")
    
    
    if __name__ == '__main__':
        unittest.main()
    

     八、web测试用例实战

    '''
    案例:百度搜索关键词"Selenium 自学网"并打开课程页面
    '''
    
    from selenium import webdriver
    import unittest
    from time import sleep
    
    
    class TestBaidu(unittest.TestCase):
        def setUp(self):
            self.driver = webdriver.Firefox()
            self.driver.implicitly_wait(10)
            self.driver.get("http://www.baidu.com")
    
        def test_baidu(self):
            driver = self.driver
            driver.find_element_by_id("kw").clear()
            driver.find_element_by_id("kw").send_keys("Selenium 自学网")
            driver.find_element_by_id("su").click()
            sleep(3)
    
            title = driver.title
            self.assertEqual(title, "Selenium 自学网 百度搜索")
    
            driver.find_element_by_partial_link_text("Selenium 自动化").click()
            sleep(5)
    
        def tearDown(self):
            driver = self.driver
            driver.quit()
    
    
    if __name__ == '__main__':
        unittest.main()
    
    import unittest
    
    test_dir = './test_case'
    
    discovery = unittest.defaultTestLoader.discover(test_dir, pattern="test*.py")
    
    if __name__ == '__main__':
        runner = unittest.TextTestRunner()
        runner.run(discovery)
    

    九、测试报告生成

    自动化测试执行完成之后,需要生存测试报告来查看测试结果,使用HTMLTestRunner模块可以直接生成html格式的报告。

    import unittest
    from HTMLTestRunner import HTMLTestRunner
    import time
    
    # 定义测试用例路径
    test_dir = './test_case'
    
    discovery = unittest.defaultTestLoader.discover(test_dir, pattern="test*.py")
    
    if __name__ == '__main__':
        # 存放报告的文件夹
        report_dir = './test_report'
        # 报告命名时间格式化
        now = time.strftime("%Y-%m-%d %H_%M_%S")
        # 报告文件完整路径
        report_name = report_dir + '/' + now + 'result.html'
    
        # 打开文件在报告文件写入测试结果
        with open(report_name, "wb") as f:
            runner = HTMLTestRunner(stream=f, title="Test Report", description="Test Case Result")
            runner.run(discovery)
    

    十、测试报告优化

    import unittest
    from BSTestRunner import BSTestRunner
    import time
    
    # 定义测试用例路径
    test_dir = './test_case'
    
    discovery = unittest.defaultTestLoader.discover(test_dir, pattern="test*.py")
    
    if __name__ == '__main__':
        # 存放报告的文件夹
        report_dir = './test_report'
        # 报告命名时间格式化
        now = time.strftime("%Y-%m-%d %H_%M_%S")
        # 报告文件完整路径
        report_name = report_dir + '/' + now + 'result.html'
    
        # 打开文件在报告文件写入测试结果
        with open(report_name, "wb") as f:
            runner = BSTestRunner(stream=f, title="Test Report", description="Test Case Result")
            runner.run(discovery)
    
  • 相关阅读:
    python 远程 部署和运行
    学习笔记——UML类图
    Core Data 多线程操作实战篇
    Core Data系列六——Custom Migration
    Core Data系列五——数据迁移方案
    NSOperation以及NSOperationQueue的使用
    Magical Record设计小谈
    Core Data系列四——多线程设计
    Core Data系列三——基本使用
    Core Data系列二——基础概念
  • 原文地址:https://www.cnblogs.com/xidian2014/p/10423182.html
Copyright © 2011-2022 走看看