python单元测试框架Unittest详解(pycharm单元测试)
ccwgpt 2024-10-25 10:47 33 浏览 0 评论
前言
我们今天来聊聊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文件后,用浏览器打开:
好了,今天的文章就到这里了,喜欢的可以点赞评论和关注哟。
相关推荐
- 用Deepseek扩写土木工程毕业论文实操指南
-
用Deepseek扩写毕业论文实操指南一、前期准备整理现有论文初稿/提纲列清楚论文核心框架(背景、现状、意义、方法、数据、结论等)梳理好关键文献,明确核心技术路线二、Deepseek扩写核心思路...
- 985学霸亲授,DeepSeek也能绘6大科研图表,5分钟就出图
-
在实验数据处理中,高效可视化是每个科研人的必修课。传统绘图软件操作复杂、耗时费力,而智能工具DeepSeek的出现彻底改变了这一现状。本文将详解如何用DeepSeek一键生成六大科研常用图表,从思维导...
- AI写论文刷屏?大学生正在丢掉的思考力
-
一、宿舍深夜:当论文变成"Ctrl+C+V"凌晨两点的大学宿舍,小王对着电脑屏幕叹气。本该三天前开始写的近代史论文,此刻还一片空白。他熟练打开某AI写作网站,输入"论五四运动的...
- Grok在辅助论文写作上能不能既“聪明”又“可怕”?!
-
AcademicIdeas-学境思源AI初稿写作随着人工智能技术的飞速发展,论文写作这一学术任务正迎来新的助力。2025年2月18日,美国xAI公司推出了备受瞩目的Grok3模型,其创始人埃隆·...
- 大四论文沟通场景!音频转文字难题听脑AI来化解
-
大四学生都知道,写论文时和导师沟通修改意见,简直是“过关斩将”。电话、语音沟通完,想把导师说的修改方向、重点要求记下来,麻烦事儿可不少。手写记不全,用普通录音转文字工具,转完还得自己慢慢找重点,稍不注...
- 论文写作 | 技术路线图怎么画?(提供经典优秀模板参考)
-
技术路线图是一种图表或文字说明,用于描述研究目标、方法和实施计划。它展示了研究的整体框架和步骤,有助于读者理解研究的逻辑和进展。在课题及论文中,技术路线图是常见的一部分,甚至是一个类似心脏一样的中枢器...
- 25年信息系统项目管理师考试第2批论文题目写作建议思路框架
-
25年信息系统项目管理师考试第2批论文题目写作建议思路框架--马军老师
- 微信购物应尽快纳入法律框架(微信购物管辖)
-
符向军近日,甘肃省工商行政管理局发布《2016年上半年信息分析报告》。报告显示,微信网购纠纷迅猛增长,网络购物投诉呈上升趋势。投诉的主要问题有出售的商品质量不过关、消费者通过微信付款后对方不发货、购买...
- 泛珠三角区域网络媒体与腾讯微信签署《战略合作框架协议》
-
新海南客户端、南海网7月14日消息(记者任桐)7月14日上午,参加第四届泛珠三角区域合作网络媒体论坛的区域网络媒体负责人及嘉宾一行到腾讯微信总部座谈交流,并签署《战略合作框架协议》(以下简称《框架协...
- 离线使用、植入微信-看乐心Mambo手环如何打破框架
-
从2014年开始智能手环就成功进入人们的生活,至今已经演变出数据监测、信息推送、心率监测等诸多五花八门的功能,人们选择智能手环并不指望其能够改变身体健康情况,更多的是通过数据来正视自身运动情况和身体健...
- 华专网络:如何零基础制作一个网站出来?
-
#如何零基础制作一个网站出来?#你是不是觉得网站建设很复杂,觉得自己是小白,需求不明确、流程搞不懂、怕被外包公司坑……这些问题我都懂!今天华专网络就用大白话给你捋清楚建站的全流程,让你轻松get网站制...
- WAIC2024丨明日上午9点,不见不散!共同探讨智能社会与全球治理框架
-
大咖云集,硕果闪耀WAIC2024世界人工智能大会智能社会论坛将于7月5日9:00-12:00与你相约直播间WAIC2024上海杨浦同济大学哔哩哔哩多平台同步直播探讨智能社会与全球治理框架WAIC...
- 约基奇:森林狼换来戈贝尔时大家都在嘲笑 他们的阵容框架很不错
-
直播吧5月4日讯西部季后赛半决赛,掘金将迎战森林狼,约基奇赛前接受采访。约基奇说道:“当蒂姆-康纳利(森林狼总经理、前掘金总经理&曾选中约基奇)做了那笔交易(换来戈贝尔)时,每个人都在嘲笑他...
- 视频号带货为什么一个流量都没有?顶级分析框架送给你
-
视频号带货为什么一个流量都没有?遇到问题,一定是步步来分析内容,视频号带货一个流量都没有,用另外一个意思来讲,就可以说是零播放。为什么视频号带货一个流量都没有?跟你说再多,都不如来个分析框架。1、是否...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- MVC框架 (46)
- spring框架 (46)
- 框架图 (58)
- flask框架 (53)
- quartz框架 (51)
- abp框架 (47)
- jpa框架 (47)
- laravel框架 (46)
- springmvc框架 (49)
- 分布式事务框架 (65)
- scrapy框架 (56)
- shiro框架 (61)
- 定时任务框架 (56)
- java日志框架 (61)
- JAVA集合框架 (47)
- grpc框架 (55)
- ppt框架 (48)
- 内联框架 (52)
- winform框架 (46)
- gui框架 (44)
- cad怎么画框架 (58)
- ps怎么画框架 (47)
- ssm框架实现登录注册 (49)
- oracle字符串长度 (48)
- oracle提交事务 (47)