• 博客园logo
  • 会员
  • 众包
  • 新闻
  • 博问
  • 闪存
  • 赞助商
  • HarmonyOS
  • Chat2DB
    • 搜索
      所有博客
    • 搜索
      当前博客
  • 写随笔 我的博客 短消息 简洁模式
    用户头像
    我的博客 我的园子 账号设置 会员中心 简洁模式 ... 退出登录
    注册 登录
花儿爷
博客园    首页    新随笔    联系   管理     
6.unittest

单元测试框架unittest

单元测试

单元测试(unit testing)是指对软件中的最小可测试单元进行检查和验证。对于单元测试中单元的含义,一般来说,要根据实际情况去判定其具体含义,如C语言中单元指一个函数,Java里单元指一个类,图形化的软件中可以指一个窗口或一个菜单等。总的来说,单元就是人为规定的最小的被测功能模块。

单元测试框架

在单元测试框架出现之前,开发人员在创建可执行测试时饱受折磨。最初的做法是在应用程序中创建一个窗口,配有"测试控制工具(harness)"。它只是一个窗口,每个测试对应一个按钮。这些测试的结果要么是一个消息框,要么是直接在窗体本身给出某种显示结果。由于每个测试都需要一个按钮,所以这些窗口很快就会变得拥挤、不可管理。

单元测试框架提供了一种统一的编程模型,可以将测试定义为一些简单的类,这些类中的方法可以调用希望测试的应用程序代码。开发人员不需要编写自己的测试控制工具;单元测试框架提供了测试运行程序(runner),只需要单击按钮就可以执行所有测试。利用单元测试框架,可以很轻松地插入、设置和分解有关测试的功能。测试失败时,测试运行程序可以提供有关失败的信息,包含任何可供利用的异常信息和堆栈跟踪。 不同编程语言有不同的单元测试框架,如Java 的Junit, TestNg, c#的 Nunit,Python的unittest,Pyunit,testtools, subunit....

单元测试框架作用

  • 提供用例组织与执行
  • 提供丰富的断言方法
  • 提供丰富的日志与测试结果

Python单元测试框架——unittest

unittest官方文档 https://docs.python.org/2.7/library/unittest.html

unittest单元测试框架不仅可以适用于单元测试,还可以适用WEB自动化测试用例的开发与执行,该测试框架可组织执行测试用例,并且提供了丰富的断言方法,判断测试用例是否通过,最终生成测试结果。

1.unittest核心属性

 

1.TestCase

一个TestCase的实例就是一个测试用例。什么是测试用例呢?就是一个完整的测试流程,包括测试前准备环境的搭建(setUp),执行测试代码(run),以及测试后环境的还原(tearDown)。元测试(unit test)的本质也就在这里,一个测试用例是一个完整的测试单元,通过运行这个测试单元,可以对某一个问题进行验证。

2.TestSuite

而多个测试用例集合在一起,就是TestSuite,而且TestSuite也可以嵌套TestSuite。 TestLoader是用来加载TestCase到TestSuite中的,其中有几个loadTestsFrom__()方法,就是从各个地方寻找TestCase,创建它们的实例,然后add到TestSuite中,再返回一个TestSuite实例。

3.TextTestRunner

TextTestRunner是来执行测试用例的,其中的run(test)会执行TestSuite/TestCase中的run(result)方法。 测试的结果会保存到TextTestResult实例中,包括运行了多少测试用例,成功了多少,失败了多少等信息。

4.Fixture

而对一个测试用例环境的搭建和销毁,是一个fixture。

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

#对Math类做单元测试
import unittest
class TestMath(unittest.TestCase):
    def setUp(self):          #前置条件
        print("start test")

    def test_add(self):
        j=Math(5,10)
        self.assertEqual(j.add(),15)   #判断结果相加等于15
        #self.assertEqual(j.add(),12)   #失败用例

    def tearDown(self):       #后置条件
        print("test end")

if __name__=='__mian__':
    #构造测试集
    suite=unittest.TestSuite()
    suite.addTest(TestMath("test_add"))

    runer=unittest.TextTestRunner
    runer.run(suite)

5.unittest.skip():装饰器,当运行用例时,有些用例可能不想执行等,可用装饰器暂时屏蔽该条测试用例。一种常见的用法就是比如说想调试某一个测试用例,想先屏蔽其他用例就可以用装饰器屏蔽。

@unittest.skip(reason): skip(reason)装饰器:无条件跳过装饰的测试,并说明跳过测试的原因。

@unittest.skipIf(reason): skipIf(condition,reason)装饰器:条件为真时,跳过装饰的测试,并说明跳过测试的原因。

@unittest.skipUnless(reason): skipUnless(condition,reason)装饰器:条件为假时,跳过装饰的测试,并说明跳过测试的原因。

@unittest.expectedFailure(): expectedFailure()测试标记为失败。

1.用setUp与setUpClass区别

setup():每个测试case运行前运行
teardown():每个测试case运行完后执行
setUpClass():必须使用@classmethod 装饰器,所有case运行前只运行一次
tearDownClass():必须使用@classmethod装饰器,所有case运行完后只运行一次

import unittest

class TestMethod(unittest.TestCase):
    @classmethod
    def setUpClass(cls):
        print("开始前执行一次---->setUpClass")
    @classmethod
    def tearDownClass(cls):
        print("结束后执行一次---->tearDownClass")
    def setUp(self):
        print("每个方法执行前执行----->setup")

    def tearDown(self):
        print("每个方法执行前执行----->teardown")

    def test_01(self):
        print("case----->01")
    def test_02(self):
        print("case----->02")

if __name__=='__main__':
    unittest.main()

verbosity使用说明
if __name__=='__main__':
    unittest.main(verbosity=1)    #默认1
    #0代表得到执行的测试总数和全局结果
    #1代表得到成功的显示,失败的显示f,错误的显示e
    #2可以得到详细的信息
2.@是修饰符,classmethod是python里的类方法
class Math:
    def __init__(self,a,b):
        self.a=int(a)
        self.b=int(b)
    def add(self):
        return self.a+self.b

#对Math类做单元测试
import unittest
class TestMath(unittest.TestCase):
    @classmethod
    def setUpClass(cls):          #所有cass运行前运行一次
        print("start")

    @classmethod    #装饰器
    def tearDownClass(cls):    #所有cass运行后运行一次
        print("end")

    def test1(self):           #注意:每条case的名字,以test开头,以数字结尾。
        j=Math(5,10)
        self.assertEqual(j.add(),15)   #判断结果相加等于15
        print("pass")
        #self.assertEqual(j.add(),12)   #失败用例

    def test2(self):
        j=Math(5,10)
        self.assertNotEqual(j.add(),10)   #判断结果不相等,不相等测试通过
        print("pass")

    def test3(self):
        j = Math(5, 10)
        self.assertTrue(j.add() > 10)    #判断结果是否True,是True则测试用例通过。
        print("pass")
    def test4(self):
        self.assertIs("abc","abc")      #判断是否一样,一样则测试通过
        print("pass")
        # self.assertIs("123",'abc')    #失败用例

    @unittest.skip("暂时跳过用例5的测试")
    def test5(self):
        self.assertIn("abc","abc123")    #判断a是否再b中,在则测试通过
        print("pass")


#如果直接运行该文件(__name__值为__main__),则执行以下语句,常用于测试脚本是否能够正常运行
if __name__=='__mian__':
    #构造测试集
    suite=unittest.TestSuite()
    suite.addTest(TestMath("test1"))
    suite.addTest(TestMath("test2"))
    suite.addTest(TestMath("test3"))
    suite.addTest(TestMath("test4"))
    suite.addTest(TestMath("test5"))


    runer=unittest.TextTestRunner
    runer.run(suite)

2.TestCase类的属性如下:

setUp():setUp()方法用于测试用例执行前的初始化工作。如测试用例中需要访问数据库,可以在setUp中建立数据库连接并进行初始化。如测试用例需要登录web,可以先实例化浏览器。

tearDown():tearDown()方法用于测试用例执行之后的善后工作。如关闭数据库连接。关闭浏览器。

assert*():一些断言方法:在执行测试用例的过程中,最终用例是否执行通过,是通过判断测试得到的实际结果和预期结果是否相等决定的。

assertEqual(a,b,[msg='测试失败时打印的信息']):断言a和b是否相等,相等则测试用例通过。

assertNotEqual(a,b,[msg='测试失败时打印的信息']):断言a和b是否相等,不相等则测试用例通过。

assertTrue(x,[msg='测试失败时打印的信息']):断言x是否True,是True则测试用例通过。

assertFalse(x,[msg='测试失败时打印的信息']):断言x是否False,是False则测试用例通过。

assertIs(a,b,[msg='测试失败时打印的信息']):断言a是否是b,是则测试用例通过。

assertNotIs(a,b,[msg='测试失败时打印的信息']):断言a是否是b,不是则测试用例通过。

assertIsNone(x,[msg='测试失败时打印的信息']):断言x是否None,是None则测试用例通过。

assertIsNotNone(x,[msg='测试失败时打印的信息']):断言x是否None,不是None则测试用例通过。

assertIn(a,b,[msg='测试失败时打印的信息']):断言a是否在b中,在b中则测试用例通过。

assertNotIn(a,b,[msg='测试失败时打印的信息']):断言a是否在b中,不在b中则测试用例通过。

assertIsInstance(a,b,[msg='测试失败时打印的信息']):断言a是是b的一个实例,是则测试用例通过。

assertNotIsInstance(a,b,[msg='测试失败时打印的信息']):断言a是是b的一个实例,不是则测试用例通过。

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

#对Math类做单元测试
import unittest
class TestMath(unittest.TestCase):
    def setUp(self):          #前置条件,执行顺序是123
        print("start")

    def test1(self):           #注意:每条case的名字,以test开头,以数字结尾。
        j=Math(5,10)
        self.assertEqual(j.add(),15)   #判断结果相加等于15
        print("pass")
        #self.assertEqual(j.add(),12)   #失败用例

    def test2(self):
        j=Math(5,10)
        self.assertNotEqual(j.add(),10)   #判断结果不相等,不相等测试通过
        print("pass")

    def test3(self):
        j = Math(5, 10)
        self.assertTrue(j.add() > 10)    #判断结果是否True,是True则测试用例通过。
        print("pass")
    def test4(self):
        self.assertIs("abc","abc")      #判断是否一样,一样则测试通过
        print("pass")
        # self.assertIs("123",'abc')    #失败用例
    def test5(self):
        self.assertIn("abc","abc123")    #判断a是否再b中,在则测试通过
        print("pass")

    def tearDown(self):       #后置条件
        print("end")

if __name__=='__mian__':
    #构造测试集
    suite=unittest.TestSuite()
    suite.addTest(TestMath("test1"))
    suite.addTest(TestMath("test2"))
    suite.addTest(TestMath("test3"))
    suite.addTest(TestMath("test4"))
    suite.addTest(TestMath("test5"))


    runer=unittest.TextTestRunner
    runer.run(suite)

案列1:

 

3.TestSuite类的属性如下:(组织用例时用)

addTest(): addTest()方法是将测试用例添加到测试套件中,如下方,是将test_baidu模块下的BaiduTest类下的test_baidu测试用例添加到测试套件。执行顺序:根据测试用例名称排序

import unittest

class F6(unittest.TestCase):
    def test_user_aa(self):
        print("添加用户")

    def test_user_ab(self):
        print("删除用户")

if __name__ == '__main__':
    suite = unittest.TestSuite
    #suite.addTest(F6)    #按照测试类批量执行测试用例
    suite.addTest(F6('test_user_aa'))
    suite.addTest(F6('test_user_bb'))   #按照测试用例执行测试用例
    unittest.TextTestRunner(verbosity=2).run(suite)

4.TextTextRunner的属性如下:(组织用例时需要用到)

run(): run()方法是运行测试套件的测试用例,入参为suite测试套件。

5.新增测试用例

class Math:
    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           #新增方法

#对Math类做单元测试
import unittest
class Test_add(unittest.TestCase):
    @classmethod
    def setUpClass(cls):          #所有cass运行前运行一次
        print("start")

    @classmethod    #装饰器
    def tearDownClass(cls):    #所有cass运行后运行一次
        print("end")

    def test1(self):           #注意:每条case的名字,以test开头,以数字结尾。
        j=Math(5,10)
        self.assertEqual(j.add(),15)   #判断结果相加等于15
        print("pass")
        #self.assertEqual(j.add(),12)   #失败用例

    def test2(self):
        j=Math(5,10)
        self.assertNotEqual(j.add(),10)   #判断结果不相等,不相等测试通过
        print("pass")

    def test3(self):
        j = Math(5, 10)
        self.assertTrue(j.add() > 10)    #判断结果是否True,是True则测试用例通过。
        print("pass")
    def test4(self):
        self.assertIs("abc","abc")      #判断是否一样,一样则测试通过
        print("pass")
        # self.assertIs("123",'abc')    #失败用例

    @unittest.skip("暂时跳过用例5的测试")
    def test5(self):
        self.assertIn("abc","abc123")    #判断a是否再b中,在则测试通过
        print("pass")
#sub方法单元测试
class Test_sub(unittest.TestCase):
    @classmethod
    def setUpClass(cls):  # 所有cass运行前运行一次
        print("start")

    @classmethod  # 装饰器
    def tearDownClass(cls):  # 所有cass运行后运行一次
        print("end")

    def test_sub1(self):
        i = Math(8, 8)
        self.assertEqual(i.sub(), 0)
        print("sub通过")

    def test_sub2(self):
        i = Math(5, 3)
        self.assertEqual(i.sub(), 2)
        print("sub2通过")


#如果直接运行该文件(__name__值为__main__),则执行以下语句,常用于测试脚本是否能够正常运行
if __name__=='__mian__':
    #构造测试集,将测试用例加载到测试套件中,按顺序执行
    suite=unittest.TestSuite()
    suite.addTest(TestMath("test1"))
    suite.addTest(TestMath("test2"))
    suite.addTest(TestMath("test3"))
    suite.addTest(TestMath("test4"))
    suite.addTest(TestMath("test5"))
    suite.addTest(TestMath("test_sub1"))
    suite.addTest(TestMath("test_sub2"))


    runer=unittest.TextTestRunner            #实例化TextTestRunner类
    runer.run(suite)          #使用run()方法运行测试套件(即运行测试套件中的所有用例)

 6.批量执行

  • 使用discover 可以一次调用多个脚本
  • test_dir 被测试脚本的路径
  • pattern 脚本名称匹配规则

discover加载到的用例是一个list集合,需要重新写入到一个list对象testcase里,这样就可以用unittest里面的TextTestRunner这里类的run方法去执行。

测试目录

 

test_01.py

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

#对Math类做单元测试
import unittest
class Test_add(unittest.TestCase):
    @classmethod
    def setUpClass(cls):          #所有cass运行前运行一次
        print("start")

    @classmethod    #装饰器
    def tearDownClass(cls):    #所有cass运行后运行一次
        print("end")

    def test1(self):           #注意:每条case的名字,以test开头,以数字结尾。
        j=Math(5,10)
        self.assertEqual(j.add(),15)   #判断结果相加等于15
        print("pass")
        #self.assertEqual(j.add(),12)   #失败用例

    def test2(self):
        j=Math(5,10)
        self.assertNotEqual(j.add(),10)   #判断结果不相等,不相等测试通过
        print("pass")

    def test3(self):
        j = Math(5, 10)
        self.assertTrue(j.add() > 10)    #判断结果是否True,是True则测试用例通过。
        print("pass")
    def test4(self):
        self.assertIs("abc","abc")      #判断是否一样,一样则测试通过
        print("pass")
        # self.assertIs("123",'abc')    #失败用例

    @unittest.skip("暂时跳过用例5的测试")
    def test5(self):
        self.assertIn("abc","abc123")    #判断a是否再b中,在则测试通过
        print("pass")

#如果直接运行该文件(__name__值为__main__),则执行以下语句,常用于测试脚本是否能够正常运行
if __name__=='__mian__':
    unittest.main()

test02.py

class Math:
    def __init__(self,a,b):
        self.a=int(a)
        self.b=int(b)
    def sub(self):
        return self.a-self.b
import unittest
class Test_sub(unittest.TestCase):
    @classmethod
    def setUpClass(cls):  # 所有cass运行前运行一次
        print("start")

    @classmethod  # 装饰器
    def tearDownClass(cls):  # 所有cass运行后运行一次
        print("end")

    def test_sub1(self):
        i = Math(8, 8)
        self.assertEqual(i.sub(), 0)
        print("sub通过")

    def test_sub2(self):
        i = Math(5, 3)
        self.assertEqual(i.sub(), 2)
        print("sub2通过")


#如果直接运行该文件(__name__值为__main__),则执行以下语句,常用于测试脚本是否能够正常运行
if __name__=='__mian__':
    unittest.main()

run_all_test.py

import unittest
import os
# 用例路径
case_path = os.path.join(os.getcwd(), "case")
# 报告存放路径
report_path = os.path.join(os.getcwd(), "report")
def all_case():

    discover = unittest.defaultTestLoader.discover(case_path,
                                                    pattern="test*.py",
                                                    top_level_dir=None)
    print(discover)
    return discover

if __name__ == "__main__":
    runner = unittest.TextTestRunner()
    import HTMLTestRunner
    report_path = "D:\\python3test\\report\\result.html"
    fp = open(report_path, "wb")
    runner = HTMLTestRunner.HTMLTestRunner(stream=fp,
                                           title=u'自动化测试报告,测试结果如下;',
                                           description=u'用例执行情况:')
    runner.run(all_case())
    fp.close()

 结果如下:

 编写测试用例注意事项:

在每条用例下面写上注释,有效查看用例

 总结:

 

posted on 2018-07-24 18:46  flowerszhao  阅读(225)  评论(0)    收藏  举报
刷新页面返回顶部
博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3