python单元测试框架Unittest详解

news2025/1/8 5:43:09

前言

我们今天来聊聊Python中的单元测试框架unittest,大家都知道在Python中比较热门的测试框架除了pytest就是unittest,我之前有讲过pytest所以今天想讲unittest。喜欢的可以点点关注哟。

Unittest详解

  Unittest是Python内部自带的一个单元测试的模块,它设计的灵感来源于Junit,具有和Junit类似的结构,有过Junit经验的朋友可以很快上手。Unittest具备完整的测试结构,支持自动化测试的执行,对测试用例集进行组织,并且提供了丰富的断言方法,最后生成测试报告。Unittest框架的初衷是用于单元测试,但也不限于此,在实际工作中,由于它强大的功能,提供的完整的测试流程,

所谓知己知彼百战不殆,首先我们来一起看下Unittest大家庭里的成员。首先导入unittest模块,使用dir()函数获取Unittest的所有成员,并输出到界面上。

1 import unittest
2 print(dir(unittest))

执行结果如下。

1 ['BaseTestSuite', 'FunctionTestCase', 'SkipTest', 'TestCase', 'TestLoader', 'TestProgram', 'TestResult', 'TestSuite', 'TextTestResult', 'TextTestRunner', '_TextTestResult', '__all__', '__builtins__', '__cached__', '__doc__', '__file__', '__loader__', '__name__', '__package__', '__path__', '__spec__', '__unittest', 'case', 'defaultTestLoader', 'expectedFailure', 'findTestCases', 'getTestCaseNames', 'installHandler', 'load_tests', 'loader', 'main', 'makeSuite', 'registerResult', 'removeHandler', 'removeResult', 'result', 'runner', 'signals', 'skip', 'skipIf', 'skipUnless', 'suite', 'util']

  结果里显示了unittest模块的各个成员,看起来非常多,不知道如何下手。其实一个模块里往往包含了大量的成员,很大一部分我们使用的频率并不高,这时需要有重点的去攻克核心的部分,其他的稍作了解即可。下面先简单介绍最常用的一些成员,后续的章节中我们会详细剖析。

(1)TestCase:可以说是unittest中最重要的一个类,也是测试用例类的父类,通过对其继承,使子类具备了执行测试的能力。下例中MainTest是需要执行的测试类。

1 class MyTest(unittest.TestCase):

(2)TestSuite: TestSuite类用于创建测试套件。最常见的用法是,使用该类将多个测试用例添加到用例集,通过运行用例集,实现多个测试用例的执行。

1 import unittest
 2 
 3 from test5 import MyTest
 4 
 5 if __name__ == '__main__':
 6     # 生成一个测试套件
 7     suite = unittest.TestSuite()
 8     # 向套件中添加要执行的测试用例
 9     suite.addTest(MyTest("test_a_eq_2"))
10     suite.addTests([MyTest("test_a_gt_1_x_le_1"), MyTest("test_a_le_1_x_gt_1")])  # addTests参数为一个列表,可以添加多个测试用例
11     # 执行套件内用例
12     unittest.main(defaultTest='suite')

3)main:调用unittest.main()方法可以方便的将测试类里的以“test”命名开头的测试方法以脚本的形式自动执行。

1 if __name__ == '__main__':
2     print('main')
3     unittest.main()
# 命令行执行测试用例  也是默认的执行方式
# python -m unittest -v test5.py
# -m表示以命令行的方式,-v表示打印详细信息
# 使用main的方式执行测试用例
# 需要配置运行方式,并且必须点击运行按钮运行,不能右键运行
配置步骤如下:

 

点击edit configuration:

 

添加完成后运行文件时选择刚刚添加的运行方式:

 

 

(4)TextTestRunner:主要使用该类的run()方法来运行TestSuite添加好的测试用例。

1 if __name__ == '__main__':
2     # 通过TestLoader加载并执行测试用例,defaultTestLoader
3     suite = unittest.defaultTestLoader.discover('./', 'test5.py')
4     # 通过unittest自带的文本运行运行测试用例
5     unittest.TextTestRunner().run(suite)

(5)skipXX:装饰器。有时我们的测试只想运行其中的一部分用例,那么我们可以使用skip装饰器来跳过执行。最常见的场景是:在不同的系统环境上运行时,某些用例是不能通过的,但这并不是我们的产品或用例导致,而是环境的不兼容等问题,此时我们可以使用skip装饰器来处理。

1     @unittest.skip("强制跳过")
1     @unittest.skipIf(3 > 2, '符合条件就跳过')
1     @unittest.skipUnless(3 > 2, '除非符合指定条件,否则跳过')
1     @unittest.expectedFailure  # 预期标记失败,如果执行时真的失败or报错,则会被认为测试通过Pass(因为

2. 重要概念

在继续学习之前,我们需要掌握四个Unittest的重要概念。以下是官方网站上通过面向对象大的方式进行的解释。

(1)test fixture:翻译过来是测试固定装置的意思。形象的说,把整个测试过程看作大的装置,这个装置里不仅具有测试执行部件,还有测试之前环境准备和测试之后环境清理的部件,有机的结合起来就是一个更大的测试装置,即test fixture。

它包括一些固定的执行方法:

1     @classmethod
 2     def setUpClass(cls) -> None:
 3         print('类前执行')
 4 
 5     @classmethod
 6     def tearDownClass(cls) -> None:
 7         print('类后执行')
 8 
 9     def setUp(self) -> None:  # 测试用例之前执行的方法
10         pass
11 
12     def tearDown(self) -> None:  # 测试用例之后执行的方法
13         pass

(2)test case:测试用例,注意与前面的TestCase类不是同一个概念。一个完整的测试流程就是一个测试用例,通过一些特定的输入得到响应,并对响应进行校验的过程。我们通过去继承TestCase这个父类,可以创建新的测试用例。

注意测试用例必须以“test_”开头

1     def test_a_eq_2(self):
2         """测试情况一"""  # 添加三引号的字符串可以在生成HTMLTestRunner测报告时增加用例注释
3         self.assertEqual(Do(2, 0, 2), 3)
4 
5     @unittest.skip("强制跳过")
6     def test_a_gt_1_x_le_1(self):
7         """测试情况二"""
8         self.assertEqual(Do(3, 0, 0.5), 1 / 6)

注意:setUpclass、tearDownclass使用是必须添加类方法装饰器@classmethod

1     @classmethod
2     def setUpClass(cls) -> None:
3         print('类前执行')
4 
5     @classmethod
6     def tearDownClass(cls) -> None:
7         print('类后执行')

 

这里-> None的部分是可以去掉的,意思是这个函数建议返回None。

(3)test suite:测试套件,也称为测试集合。多个测试用例组合在一起就形成了测试集,当然测试集里不仅能包含测试用例,也可以再次嵌套测试集,测试集可以用于代码的组织和运行。

定义测试集可以一个一个添加:

1 if __name__ == '__main__':
2     # 生成一个测试套件
3     suite = unittest.TestSuite()
4     # 向套件中添加要执行的测试用例
5     suite.addTest(MyTest("test_a_eq_2"))
6     suite.addTests([MyTest("test_a_gt_1_x_le_1"), MyTest("test_a_le_1_x_gt_1")])  # addTests参数为一个列表,可以添加多个测试用例
7     # 执行套件内用例
8     unittest.main(defaultTest='suite')

也可以通过TestLoader加载并执行测试用例,同时添加多个py文件中的测试用例:

1 if __name__ == '__main__':
2     # 通过TestLoader加载并执行测试用例,defaultTestLoader
3     suite = unittest.defaultTestLoader.discover('./', 'test5.py')
4     unittest.TextTestRunner().run(suite)

 

(4)test runner:是Unittest中的重要组成部分,主要职责为执行测试,通过图形、文本或者返回一些特殊值的方式来呈现最终的运行结果。例如执行的用例数、成功和失败的用例数。

  执行结果:

    . 表示pass,用例通过

    F 表示false,用例执行失败

    E 表示error,用例出错

    S 表示skip,用例执行跳过

下图展示了他们之间的关系,test fixture是包含了以test case为核心的整个组件,多个test case可以集合到一个test suite中,最后调用test runner执行并生成结果。

 

unittest断言方式:

assertEqual 等于 assertNotEqual

assertTrue 为True

assertIn 在..之中

一般我们使用HTMLTestRunner来生成html形式的测试报告。

使用HTMLTestRunner步骤如下:

1、在对应python文件的Lib目录下添加HTMLTestRunner.py文件

这个文件可以在:http://tungwaiyip.info/software/HTMLTestRunner.html 。这里下载:

将文件保存到python的Lib目录下后,需要修改几个地方,因为这个py文件中的一些地方可能和python3有不兼容之处:

第94行,将import StringIO修改成import io

第539行,将self.outputBuffer = StringIO.StringIO()修改成self.outputBuffer= io.StringIO()

第642行,将if not rmap.has_key(cls):修改成if not cls in rmap:

第766行,将uo = o.decode(‘latin-1‘)修改成uo = e

第775行,将ue = e.decode(‘latin-1‘)修改成ue = e

第631行,将print >> sys.stderr, ‘\nTime Elapsed: %s‘ %(self.stopTime-self.startTime)修改成print(sys.stderr, ‘\nTimeElapsed: %s‘ % (self.stopTime-self.startTime))

修改完成后,就可以使用HTMLTestRunner来生成测试报告了

1 from HTMLTestRunner import HTMLTestRunner
 2 
 3 if __name__ == '__main__':
 4     suite = unittest.defaultTestLoader.discover('./', pattern='test5.py')
 5     # 为了避免每次运行生成的报告名称都相同,可以使用时间作为文件名称的一部分
 6     now = time.strftime('%Y_%m_%d %H_%M_%S', time.localtime())
 7     file = open('reports_' + now + '.html', 'wb')
 8 
 9     # unittest.TextTestRunner().run(suite)
10     htmlrunner = HTMLTestRunner(stream=file, title='自动化测试报告', description='测试报告内容描述如下:')
11     htmlrunner.run(suite)

生成html文件后,用浏览器打开:

 

最后: 为了回馈铁杆粉丝们,我给大家整理了完整的软件测试视频学习教程,朋友们如果需要可以自行免费领取 【保证100%免费】

在这里插入图片描述

全套资料获取方式:点击下方小卡片自行领取即可

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/688410.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

mysql——初步认识

数据库是什么? 数据库(Database)是按照数据结构来组织、存储和管理数据的仓库 说人话 就是 数据库是一个电子仓库,里面存了一些数据 我们要学习的mysql就是数据库中的一种,并且是一种关系型数据库,当然有…

ASEMI快恢复二极管MUR20100DCR的性能与应用

编辑-Z 本文主要介绍了MUR20100DCR二极管的性能与应用。我们将对MUR20100DCR二极管的基本性能、不同领域的应用和优势与不足进行分析。 1、MUR20100DCR二极管的基本性能 MUR20100DCR二极管是一种高性能的超快速二极管,具有高电压、高电流和低漏电流等特点。它采用…

基于Python所写的企业编码生成系统

点击以下链接获取源码资源: https://download.csdn.net/download/qq_64505944/87950401?spm1001.2014.3001.5503 在PyCharm中运行《企业编码生成系统》即可进入如图1所示的系统主界面。在该界面中可以选择要使用功能对应的菜单进行不同的操作。在选择功能菜单时&…

那些曾经考过的turtle绘图题(11~15)

【编程实现绘图 -11】 利用random库和turtle库,在屏幕上绘制4个小雪花,雪花半径随机,坐标由列表points给出,雪花颜色为红色,效果如图所示 points = [[0,0], [50, 40], [70, 80], [-40, 30]] # 样例代码 from turtle import * # 导入turtle库 import random pensize(4) …

【深入探讨】区块链的历史与现状

发表时间:2023年5月18日 最近,BSV比特币协会在德克萨斯州奥斯汀举办了首届Unbound Perspectives Live Summit活动。本次活动采取非公开形式,大约有100人受邀参会。 与会者包括了Unbounded Capital公司的有限合伙人、知名风险投资和对冲基金经…

跳跃表详解

跳跃表[SkipList]是一种基于有序链表的扩展,简称跳表,其就是使用关键节点作为索引的一种数据结构 怎样能更快查找到一个【有序链表】的某一节点呢? 可以利用类似【索引】的思想,提取出【链表】中的【部分关键节点】 比如&#…

bean的实例化和初始化

Instantiation:表示实例化,对象还未生成 Initialization:表示初始化,对象已经生成 InstantiationAwareBeanPostProcessor继承自BeanPostProcessor,它内部提供了3个方法,再加上BeanPostProcessor接口内部的2个方法,所…

VS Code 安装

前端开发工具 VSCodeHBuildersublimeWebStorm VS Code 一、介绍 Visual Studio Code(简称 VS Code )是 Microsoft 于2015年4月发布的一款代码编辑器。VS Code 对前端代码有非常强大的支持,同时也其他编程语言(例如&#xff1a…

20年运维老兵,SRE领域大咖张观石揭秘FinOps体系实践方法

当前,降本增效成为各大互联网公司的重要方向,IT成本则占据了互联网成本的大头。随着IT资源成本花费越来越高,很多公司意识到掌握管控成本和优化成本的重要性。 如何有效的降本?如何做好成本的洞察管控?如何掌握资源成…

ECDH 算法

一、简介 ECDH全称是椭圆曲线迪菲-赫尔曼秘钥交换(Elliptic Curve Diffie–Hellman key Exchange),主要是用来在一个不安全的通道中建立起安全的共有加密资料,一般来说交换的都是私钥,这个密钥一般作为“对称加密”的…

STM32速成笔记—IWDG

文章目录 一、IWDG简介二、STM32的IWDG2.1 STM32的IWDG简介2.2 喂狗2.3 IWDG框图 三、IWDG配置步骤四、IWDG配置程序4.1 IWDG初始化程序4.2 喂狗 五、应用实例 一、IWDG简介 独立看门狗(Independent Watchdog, IWDG),什么是看门狗&#xff1…

【二叉树part06】| 654.最大二叉树、617.合并二叉树、700.二叉搜索树中的搜索、98.验证二叉搜索树

目录 🎈LeetCode654.最大二叉树 🎈LeetCode617.合并二叉树 🎈LeetCode700. 二叉搜索树中的搜索 🎈LeetCode98. 验证二叉搜索树 🎈LeetCode654.最大二叉树 链接:654.最大二叉树 给定一个不重复的整数数…

林客本地生活商城小程序开发

本地生活商城小程序的市场前景非常广阔。近年来,随着移动互联网技术的普及和人们消费观念的变化,越来越多的消费者开始通过手机进行线上购物。本地生活商城小程序恰好满足了用户对于便捷、快速、个性化购物体验的需求,具有以下优势&#xff1…

1752_使用Perl实现目录遍历

全部学习汇总: GreyZhang/speed_emacs: Try to make a new emacs configuration which is fast even on windows! (github.com) 关于Perl和Python的优劣我不去做什么分析,也不去发表什么深入的见解。我个人的学习过程是先Perl后Python再回到Perl。因为工…

【2023.6.26】记达梦数据库基于信创服务器麒麟OS报错与优化

一、场景描述 OS:银河麒麟V10(GUI)CPU:鲲鹏920达梦数据库:V8 (基于平台信创:麒麟V10、鲲鹏架构)金蝶中间件:V9 二、达梦数据库报错记录 报错1:(SWT:18564&…

基于Java削面快餐店点餐服务系统设计实现(源码+lw+部署文档+讲解等)

博主介绍: ✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战 ✌ 🍅 文末获取源码联系 🍅 👇🏻 精…

cmd中输入npm install,回车——安装node modules依赖,出现报错的【解决方法】

目录 1.正常情况是: 2.当前问题: 3.解决方法: 当拿到一个前端项目的代码文件夹的时候,想要启动项目。 如果项目的代码文件夹里面没有node modules文件夹: 需要打开cmd,然后在里面输入 npm install (可…

c++初始化列表和静态成员

文章目录 一、初始化列表二、静态(static)成员 一、初始化列表 类对于成员变量的赋值有两种方式 函数体内赋值初始化列表 函数体内赋值就是直接在构造函数中赋值即可,而初始化列表它是较为特殊的一种方式! 初始化列表是在构造函数()之后以:开始,然后后…

查询穿梭框实现

技术栈: 使用vue3 的composition api 和 tsx 进行开发 一、需求描述 点击编辑,显示穿梭框: 可将左侧维度拖动至右侧,已选维度可上下拖动调整顺序。 二、需求分析 1、状态传递过程 我们首先来分析一下,整个流程中的…

魔改xxl-job,彻底告别手动配置任务!自动注册xxljob定时任务

xxl-job是一款非常优秀的任务调度中间件,轻量级、使用简单、支持分布式等优点,让它广泛应用在我们的项目中,解决了不少定时任务的调度问题。 我们都知道,在使用过程中需要先到xxl-job的任务调度中心页面上,配置执行器…