python之单元测试框架—unittest(补充)
一. unittest最核心的四个概念
unittest中最核心的四个概念是:test case,test suite,test runner,test fixture

TestCase:一个testcase的实例就是一个测试用例:测试前准备环境的搭建(setUp),执行测试代码(run),以及测试后环境的还原(tearDown)
TestSuite:多个测试用例集合在一起
TestLoader:是用来加载TestCase到TestSuite中的
TextTestRunner:用来执行测试用例的。其中的run(test)会执行TestSuite/TestCase中的run(result)方法
TextTestResult:保存TextTestRunner执行的测试结果
fixture:测试用例环境的搭建和销毁(setUp/setUpClass,tearDown/tearDownClass)
二. unittest初级使用
1. 导入unittest模块、被测文件或者其中的类
2. 创建一个测试类,并继承unittest.TestCase
3. 重写setUp和tearDown方法(如果有初始化条件和结束条件)
4. 定义测试函数,函数以test_开头
5. 在函数体中使用断言来判断测试结果是否符合预期结果
6. 调用unittest.main()方法来运行测试用例
实例
1. 在工程下创建一个My_UnitTest的包,在这个包下面,创建一个被测对象myClass.py和一个测试用例Test_Myclass.py

2. 如果要测试数学中的加法和减法,测试对象myClass.py的内容为
class Math:
def add(self, a, b):
return a + b
def minus(self, a, b):
return a - b
3. 在测试用例中引入unittest,被测文件中的类,注意测试用例是以test_开头
import unittest
from revise.My_UnitTest.myClass import Math
class Test_MyClass(unittest.TestCase):
def setUp(self):
self.m = Math()
def tearDown(self):
pass
def test_add(self):
result = self.m.add(100, 23)
self.assertEqual(123, result)
#减法
def test_minus(self):
result = self.m.minus(235, 111)
self.assertEqual(124, result)
4. 测试用例的执行顺序是以字母a-z和数字的从小到大的顺序来排列的,所以上述两个测试用例执行的顺序是add在前,minus再后,可以改改代码验证一下:
import unittest
from revise.My_UnitTest.myClass import Math
class Test_MyClass(unittest.TestCase):
def setUp(self):
self.m = Math()
def tearDown(self):
pass
def test_add(self):
result = self.m.add(100, 23)
print("我先执行")
self.assertEqual(123, result)
#减法
def test_minus(self):
result = self.m.minus(235, 111)
print("我后执行")
self.assertEqual(124, result)
运行结果:
Testing started at 12:36 ... D:\Program\python34\python.exe "D:\Program\PyCharm 2018.1.4\helpers\pycharm\_jb_unittest_runner.py" --path D:/python_workshop/python6/revise/My_UnitTest/Test_Myclass.py Launching unittests with arguments python -m unittest D:/python_workshop/python6/revise/My_UnitTest/Test_Myclass.py in D:\python_workshop\python6\revise\My_UnitTest 我先执行 我后执行 Ran 2 tests in 0.001s OK
如果希望minus先执行,add后执行,可以给前者函数名加一个1,后者加一个2
def test_2_add(self):
result = self.m.add(100, 23)
print("我后执行")
self.assertEqual(123, result)
#减法
def test_1_minus(self):
result = self.m.minus(235, 111)
print("我先执行")
self.assertEqual(124, result)
5. 如果在setUp里实例化被测文件中的类,那么每一条测试用例都要实例化一次被测类,用setUpClass()可以只实例化一次,同理,tearDownClass也只做一次收尾工作
@classmethod
def setUpClass(cls):
cls.m = Math()
@classmethod
def tearDownClass(cls):
pass
6. 用unittest.main()的方法来运行测试用例Test_Myclass.py
import unittest
from revise.My_UnitTest.myClass import Math
class Test_MyClass(unittest.TestCase):
@classmethod
def setUpClass(cls):
cls.m = Math()
@classmethod
def tearDownClass(cls):
pass
def test_2_add(self):
result = self.m.add(100, 23)
print("我后执行")
self.assertEqual(123, result)
#减法
def test_1_minus(self):
result = self.m.minus(235, 111)
print("我先执行")
self.assertEqual(124, result)
if "__name__" == "__main__":
unittest.main()
三. 断言Assert
TestCase类提供了一系列的断言,即结果比对的函数
|
Method |
Checks that |
New in |
|
assertEqual(a, b) |
a == b |
|
|
assertNotEqual(a, b) |
a != b |
|
|
assertTrue(x) |
bool(x) is True |
|
|
assertFalse(x) |
bool(x) is False |
|
|
assertIs(a, b) |
a is b |
3.1 |
|
assertIsNot(a, b) |
a is not b |
3.1 |
|
assertIsNone(x) |
x is None |
3.1 |
|
assertIsNotNone(x) |
x is not None |
3.1 |
|
assertIn(a, b) |
a in b |
3.1 |
|
assertNotIn(a, b) |
a not in b |
3.1 |
|
assertIsInstance(a, b) |
isinstance(a, b) |
3.2 |
|
assertNotIsInstance(a, b) |
not isinstance(a, b) |
3.2 |
四. unittest进阶使用
方式一:类名("方法名")的集合
testsuite:
addTest() 添加一个测试用例
addTests([..]) 添加多个测试用例,addTests的参数是一个测试用例的集合
注意:addTests中执行用例的顺序是按添加的先后顺序进行的,如果这样添加用例可能造成断言失败:
suite.addTests([TestFileOperate("test_00_read_all"),
TestFileOperate("test_01_write_data"),
TestFileOperate("test_03_read_all"),
TestFileOperate("test_02_add_data")])
正确的做法是:
suite.addTests([TestFileOperate("test_00_read_all"),
TestFileOperate("test_01_write_data"),
TestFileOperate("test_02_add_data"),
TestFileOperate("test_03_read_all")])
s = unittest.TestSuite()
s.addTest(testStudent("test_do_homework"))
runner = unittest.TextTestRunner()
runner.run(s)
实例1:addTest的使用
在My_UnitTest包下创建一个main.py

main.py中的代码如下:
import unittest
from revise.My_UnitTest.Test_Myclass import Test_MyClass
#实例化测试套件对象
s = unittest.TestSuite()
#调用addTest来加载测试用例——addTest(类名("用例函数名称"))——添加一个测试用例
s.addTest(Test_MyClass("test_add"))
s.addTest(Test_MyClass("test_minus"))
#使用TextTestRunner来运行测试用例
#实例化
runner = unittest.TextTestRunner()
#用run方法就是用来运行测试用例的
runner.run(s)
运行结果,控制台输出如下:
D:\Program\python34\python.exe D:/python_workshop/python6/revise/My_UnitTest/main.py .. ---------------------------------------------------------------------- Ran 2 tests in 0.001s OK
实例2:addTests的使用
import unittest
from revise.My_UnitTest.Test_Myclass import Test_MyClass
#实例化测试套件对象
s = unittest.TestSuite()
#加载多个测试用例——参数为列表——列表当中为测试用例
s.addTests([Test_MyClass("test_add"), Test_MyClass("test_minus")])
#使用TextTestRunner来运行测试用例
#实例化
runner = unittest.TextTestRunner()
#用run方法就是用来运行测试用例的
runner.run(s)
运行结果:
D:\Program\python34\python.exe D:/python_workshop/python6/revise/My_UnitTest/main.py .. ---------------------------------------------------------------------- Ran 2 tests in 0.000s OK
输出测试报告-text-到文件
#创建一个文件,以写的方式打开
fs = open("test_result.txt", "w")
runner = unittest.TextTestRunner(fs)
runner.run(s)
实例3:输出测试报告到文件
import unittest
from revise.My_UnitTest.Test_Myclass import Test_MyClass
#实例化测试套件对象
s = unittest.TestSuite()
#加载多个测试用例——参数为列表——列表当中为测试用例
s.addTests([Test_MyClass("test_add"), Test_MyClass("test_minus")])
#使用TextTestRunner来运行测试用例
#打开一个文件
fs = open("test_run_result.txt", "w")
#实例化
runner = unittest.TextTestRunner(fs)
#用run方法就是用来运行测试用例的
runner.run(s)
运行结果,控制台没有输出,发现当前目录下多了一个文本文件test_run_result.txt

控制台输出的信息写到文本文件中了

方式二:unittest.TestLoader.discover方法匹配目录下的用例
假如现在目录下存在两个测试用例,Test_Myclass.py和Test)_Myclass2.py,如果用addTests的方法添加用例到测试套件,未免有点麻烦,这时候需要使用TestLoader()这个类

代码如下:
import unittest, os #实例化测试套件对象 s = unittest.TestSuite() #1.实例化TestLoader对象 2.使用discover去找到一个目录下的所有测试用例 loader = unittest.TestLoader() #3.使用addTests将找到的测试用例放在测试套件下 s.addTests(loader.discover(os.getcwd())) #运行 runner = unittest.TextTestRunner() runner.run(s)
运行结果:
.... ---------------------------------------------------------------------- Ran 4 tests in 0.000s OK
注意:源码中discover方法, start_dir是要寻找的目录路径,pattern是查找条件,即在指定的目录下查找以"test"开头的测试用例(事实上,这个查找是不区分大小写的,即Test开头的也能找到)

方式三:unittest.TestLoader.loadTestsFromModule匹配模块中的测试用例
ps:TetLoader类、TestSuite类需要先实例化再使用
五. 美化测试报告—html
python有提供第三方库支持输出的测试报告为html样式
库名:HtmlTestRunner
导入:from HtmlTestRunnerNew import HTMLTestRunner
使用方式:
s = unittest.TestSuite()
s.addTests(测试用例)
fp = open(dir_config.htmlreport_dir + "/API_autoTest_{0}.html".format(curTime), "wb")
runner = HTMLTestRunnerNew.HTMLTestRunner(
stream = fp,
title = "QCD接口测试报告",
description = "QCD接口测试报告",
tester = "xiaozhai"
)
runner.run(s)
代码修改如下:
import unittest, os, time
from HTMLTestRunnerNew import HTMLTestRunner
#实例化测试套件对象
s = unittest.TestSuite()
#1.实例化TestLoader对象 2.使用discover去找到一个目录下的所有测试用例
loader = unittest.TestLoader()
#3.使用addTests将找到的测试用例放在测试套件下
s.addTests(loader.discover(os.getcwd()))
#获取当前时间
curTime = time.strftime("%Y-%m-%d_%H-%M-%S")
#在当前目录下创建一个html文件
fp = open(os.getcwd() + "/autoTest_report_{0}.html".format(curTime), "wb")
#运行测试用例,生成测试报告
runner = HTMLTestRunner(
stream=fp,
title="单元测试报告",
description="Math类的单元测试报告",
tester="xiaozhai"
)
runner.run(s)
运行效果:
D:\Program\python34\python.exe D:/python_workshop/python6/revise/My_UnitTest/main.py ok test_add (Test_Myclass.Test_MyClass) ok test_minus (Test_Myclass.Test_MyClass) ok test_add (Test_Myclass2.Test_MyClass2) ok test_minus (Test_Myclass2.Test_MyClass2) Time Elapsed: 0:00:00.001000 Sun Jul 8 18:17:18 2018 - Start Test:test_add (Test_Myclass.Test_MyClass) Sun Jul 8 18:17:18 2018 - Start Test:test_minus (Test_Myclass.Test_MyClass) Sun Jul 8 18:17:18 2018 - Start Test:test_add (Test_Myclass2.Test_MyClass2) Sun Jul 8 18:17:18 2018 - Start Test:test_minus (Test_Myclass2.Test_MyClass2)


参考文章
https://blog.csdn.net/huilan_same/article/details/52944782
python之单元测试框架—unittest(补充)的更多相关文章
- Appium+python的单元测试框架unittest(1)(转)
unittest为python语言自带的单元测试框架,python把unittest封装为一个标准模块封装在python开发包中.unittest中常用的类有:unittest.TestCase.un ...
- Appium+python的单元测试框架unittest(4)——断言(转)
(原文:https://www.cnblogs.com/fancy0158/p/10051576.html) 在我们编写的测试用例中,测试步骤和预期结果是必不可少的.当我们运行测试用例时,得到一个运行 ...
- python之单元测试框架—unittest
一. 什么是单元测试?单元测试的对象是什么? 1: 什么是单元测试? 按照定义,单元测试就是对单个模块或者单个类或者单个函数进行测试,一般是开发做的,按照阶段分,一般就是单元测试.集成测试.系统测试. ...
- Python之单元测试框架unittest
创建class继承unittest,每一个测试用例是以test开头的函数,先执行setup,然后用例按照字母的顺序执行,然后执行teardown import unittest class demo( ...
- Appium+python的单元测试框架unittest(3)——discover(转)
(原文:https://www.cnblogs.com/fancy0158/p/10047906.html) TestSuite套件可以添加很多个用例后运行,但是每个用例都需要调用addTest()函 ...
- Appium+python的单元测试框架unittest(2)——fixtures(转)
(原文:https://www.cnblogs.com/fancy0158/p/10046333.html) unittest提供的Fixtures用以在测试执行前和执行后进行必要的准备和清理工作,可 ...
- Python单元测试框架unittest之单用例管理(一)
一.概述 本文介绍python的单元测试框架unittest,unittest原名为PyUnit,是由java的JUnit衍生而来,这是Python自带的标准模块unittest.unittest是基 ...
- 单元测试框架unittest
单元测试:单元测试,是指对软件中的最小可测试单元进行检查和验证,对于单元测试中单元的含义,一般来说,要根据实际情况去判定其具体含义如:c语言中单元指一个函数,java里单元指一个类,图形化的软件中可以 ...
- Python单元测试框架unittest使用方法讲解
这篇文章主要介绍了Python单元测试框架unittest使用方法讲解,本文讲解了unittest概述.命令行接口.测试案例自动搜索.创建测试代码.构建测试套件方法等内容,需要的朋友可以参考下 概 ...
随机推荐
- swiper插件简介及用法
swiper Swiper是纯javascript打造的滑动特效插件,面向手机.平板电脑等移动终端.Swiper能实现触屏焦点图.触屏Tab切换.触屏多图切换等常用效果.Swiper开源.免费.稳定. ...
- (转) RabbitMQ学习之延时队列
http://blog.csdn.net/zhu_tianwei/article/details/53563311 在实际的业务中我们会遇见生产者产生的消息,不立即消费,而是延时一段时间在消费.Rab ...
- 谷歌公布全新设计语言:跟苹果Swift天差地别
今日凌晨.谷歌(微博)在I/O大会上公布了全新设计语言Material Design.在20多天前的WWDC上.苹果也公布了全新编程语言Swift.两家科技巨头公司,在一年一度的开发人员大会上,都公布 ...
- (转)java类到底是如何加载并初始化的?
Java虚拟机如何把编译好的.class文件加载到虚拟机里面?加载之后如何初始化类?静态类变量和实例类变量的初始化过程是否相同,分别是如何初始化的呢?这篇文章就 是解决上面3个问题的. 若有不正之处, ...
- 自己定义图片的progressbar
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/zpf8861/article/details/36183455 Android系统自带的Progre ...
- 0502-Hystrix保护应用-简介,使用,健康指标等
一.概述 参看地址:https://cloud.spring.io/spring-cloud-static/Edgware.SR3/single/spring-cloud.html#_circuit_ ...
- django 快速数据库操作,不用SQL语句
配置models文件 # -*- coding: utf-8 -*- from __future__ import unicode_literals from django.db import mod ...
- [转]c# 画图中bitmap类处理出图片时,存储的注意事项
今天查找以前写的画图程序,想完善一下,发现 图片添加文字水印时候会有些模糊,特别是小字体的时候特别模糊, 经过一番调适,终于发现了问题 帖上代码,警示自己 System.Drawing.Image i ...
- Python(文件处理)
二.基本操作 #r''------------------>> r:原生字符串,不判断符号的含义#文件处理 f=open(r’c:\a.txt’,’r’,encoding=’utf-8’) ...
- Spring.Net依赖注入(属性注入)
一.前言: Spring.Net是Java开源框架迁移过来的,主要分为 1)依赖注入 2)面向方面编程 3)数据访问抽象 4)Asp.Net扩展 四个模块功能,这里只是简单介绍依赖注入模块功能. 对于 ...