【Python自动化测试】:Unittest单元测试与HTMLTestRunner自动生成测试用例的好帮手

读者大大们好呀!!!☀️☀️☀️


博客头像
🔥 欢迎来到我的博客
👀期待大大的关注哦❗️❗️❗️
🚀欢迎收看我的主页文章➡️寻至善的主页

文章目录

  • 🔥前言
  • 🚀unittest编写测试用例
  • 🚀unittest测试用例执行
  • 🚀unittest常见的断言方法
  • 🚀unittest测试结果分析
  • 🚀unittest测试用例的执行顺序
  • 🚀跨文件组织测试用例
  • 🚀HTMLTestRunner生成测试报告
  • ⭐️⭐️⭐️总结

🔥前言

哲学三问:什么是Unittest?Unittest可以做什么?为什么用Unitest?

1️⃣Python自带的单元测试框架,此外基于Python还有其他的单元测试框架:pytest,doctest,nose等
2️⃣编写规范的测试用例,组织测试用例,生成测试结果
3️⃣自动化编写脚本(自动化测试用例)通常使用单元测试框架来编写,组织和生成测试结果
下面就是实操环节了,尽情期待吧!

🚀unittest编写测试用例

第一步:打开你已经装好的神器:pyCharm,没错就是这个东西:
在这里插入图片描述
第二步:新建一个工程---->unitTest1
在这里插入图片描述
第三步:建立一个简单的被测试文件(包含了加减乘除的函数类)---->count.py

class Count:def __init__(self, a, b):self.a = aself.b = bdef add(self):c = self.a + self.breturn cdef sub(self):d = self.a - self.breturn ddef div(self):e = self.a * self.breturn edef mul(self):e = self.a / self.breturn e

在这里插入图片描述
第四步:根据被测函数使用unittest编写测试代码创建测试文件----Testcount.py
1️⃣这里记得选Python unit Test创建Python测试文件:
在这里插入图片描述
2️⃣随后会自动生成这些代码:

import unittestclass MyTestCase(unittest.TestCase):def test_something(self):self.assertEqual(True, False)  # add assertion hereif __name__ == '__main__':unittest.main()

在这里插入图片描述
第五步:开始编写构造测试用例函数了:

import unittest
from count import Countclass TestCaseCount(unittest.TestCase):def setUp(self) -> None: # 每个测试用例开始前执行print("这是执行的测试准备阶段!我要开始测试了!")  # add assertion heredef test_add1(self):  # 定义测试步骤与断言print("我执行的是加法函数测试!")   # 添加该print语句帮助我们了解test_add1何时被执行c1 = Count(1, 2)  # 根据Count类生成对象c1,会自动调用Count类里的init方法r1 = c1.add()  # r1保存的是实际被测代码的运行结果self.assertEqual(r1, 3)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def test_sub2(self):  # 定义测试步骤与断言print("我执行的是减法函数测试!")   # 添加该print语句帮助我们了解test_sub2何时被执行d1 = Count(2, 1)  # 根据Count类生成对象d1,会自动调用Count类里的init方法r2 = d1.sub()  # r2保存的是实际被测代码的运行结果self.assertEqual(r2, 1)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def test_div3(self):  # 定义测试步骤与断言print("我执行的是乘法函数测试!")   # 添加该print语句帮助我们了解test_div3何时被执行e1 = Count(2, 2)  # 根据Count类生成对象e1,会自动调用Count类里的init方法r3 = e1.div()  # r3保存的是实际被测代码的运行结果self.assertEqual(r3, 4)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def test_mul4(self):  # 定义测试步骤与断言print("我执行的是除法函数测试!")   # 添加该print语句帮助我们了解test_mul4何时被执行f1 = Count(4, 2)  # 根据Count类生成对象c1,会自动调用Count类里的init方法r4 = f1.mul()  # r1保存的是实际被测代码的运行结果self.assertEqual(r4, 2)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def tearDown(self) -> None:  # 每个测试用例结束后执行print("这是执行测试的结束,收拾残局阶段!!")if __name__ == '__main__':unittest.main()

PS:执行Python文件的内部调用与外部调用
在这里插入图片描述

🚀unittest测试用例执行

上面的测试用例函数结构是:
方法名必须是以“test_”打头,然后再是定义测试步骤与断言,断言会在下面讲到。现在是运行截图:
在这里插入图片描述

🚀unittest常见的断言方法

1️⃣assertEqual(a,b):判断a,b是否相等,如果相等,测试通过,如果不相等,测试失败
2️⃣assertNotEqual(a,b):判断a,b是否不相等,如果相等,测试失败

3️⃣assertTrue(x):用于判断bool(x)是否是True,如果不是True,测试失败 与1️⃣等效
4️⃣assertFalse(x):用于判断bool(x)是否是False,如果不是False,测试失败 与2️⃣等效
比如一个函数判断一个数是否是素数便可用3、4的断言方法

5️⃣assertIn(a,b):用于判断a是否在b中,如果a不在b中,则测试失败
6️⃣assertNotIn(a,b)和上个结果相反

🚀unittest测试结果分析

测试结果有三种:
1️⃣测试通过–》 . 一个点表示一个用例通过,上面的图片已经指出,下面着重介绍另外两种
2️⃣测试失败–》
测试的函数方法与预期结果不符合便是测试失败;比如我把加法的方法改一下:

    def add(self):c = self.areturn c

然后再进行测试:
在这里插入图片描述
显示三个用例成功,一个用例失败。
3️⃣测试错误–》
被测试代码或语法原因产生的错误:
现在我把被测试函数的默认属性方法改错:

# def __init__(self, a, b):def __int__(self, a, b):self.a = aself.b = b

执行后直接报错:
在这里插入图片描述

🚀unittest测试用例的执行顺序

测试方法的执行跟编写的顺序无关,而是根据test_后面接的ASCII码的顺序来执行(0-9,A-Z, _, a-z编码越小执行的优先级越高)

当然我们也可以按照自定义的顺序来执行:
不使用unittest的main方法默认顺序执行,而是通过改为使用unittest提供的TestSuit(测试套件)+TestRunner(测试运行器)的方式可以实现自定义顺序执行测试方法
我把下面main方法改一下:

if __name__ == '__main__':# unittest.main()   # 调用main()使用的是默认执行顺序# 使用test suite+test runner来自定义执行顺序# 一、按自定义顺序加载测试方法到测试套件里ts = unittest.TestSuite()  # 调用unittest提供的TestSuite类生成对象tsts.addTest(TestCaseCount("test_mul4"))# 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_mul4测试方法ts.addTest(TestCaseCount("test_div3"))# 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_div3测试方法ts.addTest(TestCaseCount("test_sub2"))# 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_sub2测试方法ts.addTest(TestCaseCount("test_add1"))# 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_add1测试方法# 二、使用测试运行器执行测试套件里的用例,生成测试结果tr = unittest.TextTestRunner()  # 调用unittest提供的TextTestRunner类生成对象trtr.run(ts)  # 调用TextTestRunner里的run方法来执行测试套件ts里的用例执行并生成结果

执行结果如下:
在这里插入图片描述

🚀跨文件组织测试用例

被测功能点肯定不止一个,而是多个,当被测功能点不断扩展,相应的测试代码也会不断的增多,这些测试代码不可能全部写在一个文件里。此时,通常的处理办法:根据所测功能点的不同,将相应的测试代码分散在不同的文件里,然后组织这些分散在不同文件里的代码一起执行。

组织分散在不同文件里的测试代码一起执行的常见办法:
🅰️Test Suite+Test Runner
🅱️discover
下面我将创建一个测试文件testcount2,里面的测试用例为其他测试数,两个runtest测试文件来区分上面🅰️,🅱️两种不同的执行方法
testcount2的文件代码为:

import unittest
from count import Countclass TestCaseCount(unittest.TestCase):def setUp(self) -> None: # 每个测试用例开始前执行print("这是第二次执行的测试准备阶段!我要开始测试了!")  # add assertion heredef test_add1(self):  # 定义测试步骤与断言print("我执行的是第二次加法函数测试!")   # 添加该print语句帮助我们了解test_add1何时被执行c1 = Count(2, 2)  # 根据Count类生成对象c1,会自动调用Count类里的init方法r1 = c1.add()  # r1保存的是实际被测代码的运行结果self.assertEqual(r1, 4)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def test_sub2(self):  # 定义测试步骤与断言print("我执行的是第二次减法函数测试!")   # 添加该print语句帮助我们了解test_sub2何时被执行d1 = Count(4, 1)  # 根据Count类生成对象d1,会自动调用Count类里的init方法r2 = d1.sub()  # r2保存的是实际被测代码的运行结果self.assertEqual(r2, 3)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def test_div3(self):  # 定义测试步骤与断言print("我执行的是第二次乘法函数测试!")   # 添加该print语句帮助我们了解test_div3何时被执行e1 = Count(1, 2)  # 根据Count类生成对象e1,会自动调用Count类里的init方法r3 = e1.div()  # r3保存的是实际被测代码的运行结果self.assertEqual(r3, 2)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def test_mul4(self):  # 定义测试步骤与断言print("我执行的是第二次除法函数测试!")   # 添加该print语句帮助我们了解test_mul4何时被执行f1 = Count(1, 1)  # 根据Count类生成对象c1,会自动调用Count类里的init方法r4 = f1.mul()  # r1保存的是实际被测代码的运行结果self.assertEqual(r4, 1)  # 将实际结果跟预期结果做等值比较,相等测试通过,不等测试失败def tearDown(self) -> None:  # 每个测试用例结束后执行print("这是第二次执行测试的结束,收拾残局阶段!!")if __name__ == '__main__':unittest.main()

runtest1的代码为:所使用的是Test Suite+Test Runner方法

import unittest
import testcount
import testcount2if __name__ == '__main__':# unittest.main()   # 调用main()使用的是默认执行顺序# 使用test suite+test runner来自定义执行顺序# 一、按自定义顺序加载测试方法到测试套件里ts = unittest.TestSuite()  # 调用unittest提供的TestSuite类生成对象tsts.addTest(testcount.TestCaseCount("test_mul4"))  # 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_mul4测试方法ts.addTest(testcount2.TestCaseCount2("test_mul4"))ts.addTest(testcount.TestCaseCount("test_div3")) # 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_div3测试方法ts.addTest(testcount2.TestCaseCount2("test_div3"))ts.addTest(testcount.TestCaseCount("test_sub2"))  # 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_sub2测试方法ts.addTest(testcount2.TestCaseCount2("test_sub2"))ts.addTest(testcount.TestCaseCount("test_add1"))  # 调用TestSuite里的addTest方法来加载TestCaseCount类里的test_add1测试方法ts.addTest(testcount2.TestCaseCount2("test_add1"))# 二、使用测试运行器执行测试套件里的用例,生成测试结果tr = unittest.TextTestRunner()  # 调用unittest提供的TextTestRunner类生成对象trtr.run(ts)  # 调用TextTestRunner里的run方法来执行测试套件ts里的用例执行并生成结果

运行结果截图:
在这里插入图片描述

runtest2的代码为:所使用的是discover方法

import unittestif __name__ == '__main__':# unittest.main()# 使用discover加载testcount.py和testcount2.py里的用例一起执行# 自动发现unitTest1包下的testcount打头的Python文件里的测试方法,将其加载到测试套件ts里ts = unittest.defaultTestLoader.discover("./", pattern="testcount*.py")# 运行器运行套件用例tr =unittest.TextTestRunner()tr.run(ts)

运行截图如下:
在这里插入图片描述

🚀HTMLTestRunner生成测试报告

既然我们做的是测试,那么便要实现生成独立的测试报告:这里我们需要用到第三方库HTMLTestRunner.py------>独立生成HTML文件格式的测试文档:
在网上(包括在Python官方文库)找不到HTMLTestRunner相关解释资料。其实HTMLTestRunner是一个第三方的unittest HTML报告库,关于unittest在Python官方文库上很容易找到:然后下载下来放到Python的lib文件下

具体实现代码在runtest2下,代码修改如下:

import unittest
import HTMLTestRunnerif __name__ == '__main__':# unittest.main()# 使用discover加载testcount.py和testcount2.py里的用例一起执行# 自动发现unitTest1包下的testcount打头的Python文件里的测试方法,将其加载到测试套件ts里ts = unittest.defaultTestLoader.discover("./", pattern="testcount*.py")# 运行器运行套件用例# tr =unittest.TextTestRunner()# tr.run(ts)# 除了可以使用unittest提供的TextTestRunner文本测试运行器来执行用例生成测试结果以外,# 还可以使用HTMLTestRunner.py模块提供的HTMLTestRunner类来执行用例生成独立的测试报告# 以二进制写的模式打开当前目录下的report.html文件,准备往里面写内容,如果文件不存在则自动创建f = open('./report.html', 'wb')tr = HTMLTestRunner.HTMLTestRunner(stream=f, title="四则运算测试报告", description="说明信息")tr.run(ts)f.close()

生成的HTML文件测试报告截图如下:
在这里插入图片描述

⭐️⭐️⭐️总结

test case测试用例unittest提供了TestCase类用来编写测试用例
test suite测试套件unittest提供了TestSuite类用来组装测试用例生成测试用例集合
test runner测试运行器unittest提供了TextTestRunner类用来执行测试用例生成测试结果
test fixture测试固件unittest提供了一系列的固件:setUp,tearDown就是测试固件的一种,用来完成测试前的准备工作和测试后的清理工作。

✈️✈️✈️如果喜欢这篇文章的话

🙏大大们可以动动发财的小手:
👉👉👉 点赞:👍收藏:⭐️评论:✍️👈👈👈

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

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

相关文章

六种常用设计模式

单例设计模式 单例模式指在整个系统生命周期里,保证一个类只能产生一个实例,确保该类的唯一性。 单例模式分类 单例模式可以分为懒汉式和饿汉式,两者之间的区别在于创建实例的时间不同: 懒汉式:指系统运行中&#…

SpringBootWeb 篇-深入了解 Mybatis 删除、新增、更新、查询的基础操作与 SQL 预编译解决 SQL 注入问题

🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 Mybatis 的基础操作 2.0 基础操作 - 环境准备 3.0 基础操作 - 删除操作 3.1 SQL 预编译 3.2 SQL 预编译的优势 3.3 参数占位符 4.0 基础操作 - 新增 4.1 主键返回…

Python图像处理:从基础到高级的全方位指南

目录 第一部分:Python图像处理基础 1.1 图像处理概念 1.2 Python图像处理常用库 1.3 实战案例:图像显示与保存 1.4 注意事项 第二部分:Python图像处理高级技巧 2.1 图像变换 2.2 图像增强 2.3 图像复原 第三部分:Python…

esp32s3中ap与sta模式的wps配对问题

无线路由器中的WPS是Wi-Fi Protected Setup的简称,中文翻译为Wi-Fi安全防护设置,它是由Wi-Fi安全联盟推出的一种无线加密认证方式。主要是为了简化无线局域网的安装及安全性能配置工作,通过这种设置,让无线连接更加方便和安全。省…

20232802 黄千里 2023-2024-2 《网络攻防实践》实践十一报告

20232802 2023-2024-2 《网络攻防实践》实践十一报告 1.实践过程 1.1web浏览器渗透攻击 攻击机:kali172.20.10.10靶机:win2k172.20.10.3 首先在kali中启动msfconsole 输入命令search MS06-014,搜索渗透攻击模块 输入use exploit/window…

终于让我找到了,你也可以学会的人工智能-机器学习教程

给大家分享一套非常棒的python机器学习课程——《AI小天才:让小学生轻松掌握机器学习》,2024年5月完结新课,提供配套的代码笔记软件包下载!学完本课程,可以轻松掌握机器学习的全面应用,复杂特征工程&#x…

C# 跨线程访问UI组件,serialPort1串口接收数据

在Windows应用程序(例如WinForms或WPF)中,UI组件(如按钮、文本框等)都在主线程(也称为UI线程)上运行。当你在一个非UI线程(例如,一个后台线程或者网络请求线程&#xff0…

关于新配置的adb,设备管理器找不到此设备问题

上面页面中一开始没有找到此android设备, 可能是因为我重新配置的adb和设备驱动, 只把adb配置了环境变量,驱动没有更新到电脑中, 点击添加驱动, 选择路径,我安装时都放在了SDK下面,可以尝试…

SpringBoot 实现 RAS+AES 自动接口解密

一、讲个事故 接口安全老生常谈了 过年之前做了过一款飞机大战的H5小游戏,里面无限模式-需要保存用户的积分,因为使用的Body传参,参数是可见的。 为了接口安全我,我和前端约定了传递参数是:用户无限模式的积分“我们…

HTML静态网页成品作业(HTML+CSS)——魅族商城首页网页(1个页面)

🎉不定期分享源码,关注不丢失哦 文章目录 一、作品介绍二、作品演示三、代码目录四、网站代码HTML部分代码 五、源码获取 一、作品介绍 🏷️本套采用HTMLCSS,未使用Javacsript代码,共有1个页面。 二、作品演示 三、代…

基于Python+OpenCV卷积神经网络的字符识别

欢迎大家点赞、收藏、关注、评论啦 ,由于篇幅有限,只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景与意义 字符识别是计算机视觉和模式识别领域的一个重要应用,它在文档数字化、车牌识别、验…

gpt-4o考场安排

说明 :经过多次交互,前后花了几个小时,总算完成了基本功能。如果做到按不同层次分配考场,一键出打印结果就完美了。如果不想看中间“艰苦”的过程,请直接跳到“最后结果”及“食用方法”。中间过程还省略了一部分交互&…

Android-多个tv_item_[i] 点击事件简写

private TextView[] tvConstellations new TextView[12];//获取当前id元素并在其点击的时候修改其颜色 for (int i 0; i < 12; i) {int resId getResources().getIdentifier("tv_constellation_" (i1), "id", getPackageName());tvConstellations[i…

神经网络与深度学习 课程复习总结

神经网络的发展历史 第一代神经网络&#xff08;1958~1969&#xff09; MCP模型&#xff08;1943年&#xff09;&#xff1a;麦卡洛克和皮茨提出了第一个人工神经元模型&#xff0c;简化为输入信号线性加权、求和、非线性激活&#xff08;阈值法&#xff09;。感知器&#xf…

鸿蒙开发 组件之间的传值

1.Prop&#xff1a;父组件传递给子组件&#xff0c;单向传递&#xff0c;子组件改变值&#xff0c;父组件UI不更新。 引入子组件 并赋值&#xff0c;子组件用Prop 接收 import headerView from ../../common/bean/BaseNavHeaderView headerView({titlestr:添加地址,isback…

go slice 扩容

扩容 slice 会迁移到新的内存位置&#xff0c;新底层数组的长度也会增加&#xff0c;这样就可以放置新增的元素。同时&#xff0c;为了应对未来可能再次发生的 append 操作&#xff0c;新的底层数组的长度&#xff0c;也就是新 slice 的容量是留了一定的 buffer 的。否则&…

【C++】STL快速入门基础

文章目录 STL&#xff08;Standard Template Library&#xff09;1、一般介绍2、STL的六大组件2.1、STL容器2.2、STL迭代器2.3、相关容器的函数vectorpairstringqueuepriority_queuestackdequeset, map, multiset, multimapunordered_set, unordered_map, unordered_multiset, …

LabVIEW2022安装教程指南【附安装包】

文章目录 前言一、安装指南1、软件包获取 二、安装步骤总结 前言 LabVIEW是一种程序开发环境&#xff0c;提供一种图形化编程方法&#xff0c;可可视化应用程序的各个方面&#xff0c;包括硬件配置、测量数据和调试&#xff0c;同时可以通过FPGA数学和分析选板中的NI浮点库链接…

有趣的css - 两个圆形加载效果

大家好&#xff0c;我是 Just&#xff0c;这里是「设计师工作日常」&#xff0c;今天分享的是一款小清新的加载动画&#xff0c;适用于 app 列表加载&#xff0c;页面加载或者弹层内容延迟加载等场景。 最新文章通过公众号「设计师工作日常」发布。 目录 整体效果核心代码html…

AWS安全性身份和合规性之Amazon Macie

Amazon Macie是一项数据安全和数据隐私服务&#xff0c;它利用机器学习&#xff08;ML&#xff09;和模式匹配来发现和保护敏感数据。可帮助客户发现、分类和保护其敏感数据&#xff0c;以及监控其数据存储库的安全性。 应用场景&#xff1a; 敏感数据发现 一家金融服务公司…