zoukankan      html  css  js  c++  java
  • 单元测试框架处理多组数据的另一种写法:基于构造函数和超继承

    众所周知,在单元测试框架中引入ddt,可以将一组数据分解为多组数据,从而实现一条数据对应一个测试用例。但是除此之外,有没有别的办法来实现这个目的呢?

    一. 代码部分

    1. 创建一个被测函数math_method.py

    class MethMethod():
        def add_test(self,a,b):
            return (a+b)

    2. 创建测试用例test_math_method.py

    import unittest
    from homework.math_method import MethMethod
    
    
    #测试类
    
    class TestMathMethod(unittest.TestCase):
    
        def __init__(self, methodName, a, b, expected):
            super(TestMethMethod, self).__init__(methodName)
            self.a = a
            self.b = b
            self.expected = expected
    
        #测试用例
    
        def test_add(self):
            try:
    
                self.assertEqual(result, self.expected)
            except Exception:
                print("出错了")
            else:
                print("计算结果是:{}".format(result))

    3. 创建测试数据test_data.txt

    [{"a":2,"b":3,"expected":5},{"a":-1,"b":-2,"expected":-3},{"a":3,"b":-1,"expected":-2},{"a":-3,"b":-1,"expected":-4}]

    4. 创建测试集test_suite.py

    import unittest
    from homework.test_math_method import TestMathMethod
    import HTMLTestRunnerNew
    
    fs= open("test_data.txt")
    datas = fs.read()
    datas = eval(datas)
    print(type(datas))
    
    
    if __name__ == '__main__':
    
        suite=unittest.TestSuite()
        for item in datas:
            suite.addTest(TestMathMethod("test_add", item['a'], item['b'], item['expected']))
        file=open("test_report.html","wb+")
        runner=HTMLTestRunnerNew.HTMLTestRunner(file,title="加法单元测试报告",tester="july")
        runner.run(suite)

    二. 代码分析

    1. 测试用例类中需要定义一个构造函数,这个构造函数继承了父类的属性,也就是unittest.TestCase的methodName参数

    同时,在子类中也重写了构造函数,添加了参数a, b, expected,而这三个则是测试用例函数test_add需要用到的

    class TestMethMethod(unittest.TestCase):
    
        def __init__(self, methodName, a, b, expected):
            super(TestMethMethod, self).__init__(methodName)
            self.a = a
            self.b = b
            self.expected = expected

    请看TestCase的源码

    class TestCase(object):

    failureException = AssertionError longMessage = True maxDiff = 80*8 # If a string is longer than _diffThreshold, use normal comparison instead # of difflib. See #11763. _diffThreshold = 2**16 # Attribute used by TestSuite for classSetUp _classSetupFailed = False def __init__(self, methodName='runTest'): """Create an instance of the class that will use the named test method when executed. Raises a ValueError if the instance does not have a method with the specified name. """ self._testMethodName = methodName self._outcome = None self._testMethodDoc = 'No test' try: testMethod = getattr(self, methodName) except AttributeError: if methodName != 'runTest': # we allow instantiation with no explicit method name # but not an *incorrect* or missing method name raise ValueError("no such test method in %s: %s" % (self.__class__, methodName)) else: self._testMethodDoc = testMethod.__doc__ self._cleanups = [] self._subtest = None

    2. 此时回到测试集中,看下面一段,TestMathMethod类中本来只有一个参数,methodName,也就是测试用例名,但此时的TestMathMethod已经被我们改写了:不仅继承了父类的methodName方法,而且构造函数中新增了三个参数a, b, expected,这样我们在收集测试用例时,可以为TestMathMethod类添加四个参数:methodName, a, b, expected。通过for循环遍历一组数据,这样可以将多组数据作为多个测试用例添加到测试集中

    for item in datas:
            suite.addTest(TestMathMethod("test_add", item['a'], item['b'], item['expected']))

    三. 比较

    ddt的方法:简单,因为涉及到装饰器,不易理解

    构造函数:略显繁琐,但容易理解

  • 相关阅读:
    join
    PS1-4
    tftp + bras
    awk调用shell
    curl
    ssh
    查看cp进度,使用watch
    tftp
    scp 链接文件的问题 + tar
    mysql必知必会(三、使用mysql)
  • 原文地址:https://www.cnblogs.com/my_captain/p/9508358.html
Copyright © 2011-2022 走看看