Python自动化测试框架之unittest使用详解!

news2024/12/28 20:06:18

这篇文章主要介绍了Python接口自动化浅析unittest单元测试原理,文中描述了单元测试,unittest模块特性、大致流程、源码及实战例子这几个模块,有需要的朋友可以借鉴参考下

以下主要介绍unittest特性、运行流程及实际案例。

一、单元测试三连问

1、什么是单元测试?

按照阶段来分,一般就是单元测试,集成测试,系统测试,验收测试。

单元测试是对单个模块、单个类或者单个函数进行测试。

将访问接口的过程封装在函数里面;

接口测试就变成了单元测试;

单元测试就是通过传参,对某个模块、某个类、某个函数进行结果输出后验证的测试。

2、为什么要做单元测试?

1.单元测试之后,才是集成测试,单个的功能模块测试通过之后,才能把单个功能模块集成起来做集成测试,为了从底层发现bug,减少合成后出现的问题。

2.越早发现bug越好,否则问题累计到后期,如果做错了就要推倒重来,对于时间和人力成本来说非常耗费精力。

对于我们测试来说:单元测试是为了执行测试用例。

3、怎么做单元测试?

Python里有两个单元测试类:

1.Unittest(Python自带);

2.Pytest(下载安装);

前者多用于接口自动化项目用,后者多用于WEB自动化项目、APP自动化项目。

二、unittest模块说明

1、unittest简介

unittest是Python自带的单元测试框,具备编写用例、组织用例、执行用例、输出报告等自动化框架的条件,可以用来作自动化测试框架的用例组织执行框架。

unittest框架的特性:

  • 提供用例组织与执行:当测试用例只有几条的时候可以不考虑用例的组织,但是当测试用例数量较多时,此时就需要考虑用例的规范与组织问题。unittest单元测试框架就是用来解决这个问题的。
  • 提供丰富的断言方法:既然是测试,就有一个预期结果和实际结果的比较问题。比较就是通过断言来实现,unittest单元测试框架提供了丰富的断言方法,通过捕获返回值,并且与预期值进行比较,从而得出测试通过与否。
  • 提供丰富的日志:每一个失败用例我们都希望知道失败的原因,所有用例执行结束我们有希望知道整体执行情况,比如总体执行时间,失败用例数,成功用例数。unittest单元测试框架为我们提供了这些数据。

2、unittest组成

unittest单元测试中最核心的四个部分是:

TestCase(测试用例)

TestSuite(测试套件)

TestRunner(测试运行器)

TestFixture(测试环境数据准备和清理)

现在我也找了很多测试的朋友,做了一个分享技术的交流群,共享了很多我们收集的技术文档和视频教程。
如果你不想再体验自学时找不到资源,没人解答问题,坚持几天便放弃的感受
可以加入我们一起交流。而且还有很多在自动化,性能,安全,测试开发等等方面有一定建树的技术大牛
分享他们的经验,还会分享很多直播讲座和技术沙龙
可以免费学习!划重点!开源的!!!
qq群号:110685036

1.TestCase(测试用例):

一个TestCase的实例就是一个测试用例。什么是测试用例呢?就是一个完整的测试流程。包括测试前准备环境的搭建(setUp)、实现测试过程的代码(run),以及测试后环境的还原(tearDown)。

单元测试(Unittest)的本质也就在这里,一个测试用例就是一个完整的测试单元,通过运行这个测试单元,可以对某一个功能进行验证。

2.TestSuite(测试套件):

一个功能的验证往往需要多个测试用例,可以把多个测试用例集合在一起执行,这就产生了测试套件TestSuite的概念。TestSuite用来组装单个测试用例。可以通过addTest加载TestCase到TestSuite中,从而返回一个TestSuite实例。

而且TestSuite也可以嵌套TestSuite。

3.TestLoader(测试用例加载器):

用来加载TestCase到TestSuite中的,其中loadTestsFrom__()方法用于寻找TestCase,并创建它们的实例,然后添加到TestSuite中,返回TestSuite实例;

4.TextTestRunner(执行测试用例):

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

5.Test Fixture(测试环境数据准备和清理):

一个测试用例的初始化准备及环境还原,主要是setUp() 和 tearDown()方法;

比如说在测试用例中需要访问数据库,那么可以在setUp()中建立数据库连接以及进行一些初始化,

在tearDown()中清除在数据库中产生的数据,然后关闭连接。

注意tearDown的过程很重要,要为以后的TestCase留下一个干净的环境。

3、unittest核心工作原理

unittest的静态类图:

大体流程:编写TestCase,由TestLoader加载TestCase到TestSuite,然后由TextTestRunner来运行TestSuite,最后将运行的结果保存在TextTestResult中。

三、unittest单元测试

1、实现思路

1.导入unittest模块、 被测文件或者其中的类;

2.创建一个测试类,并继承unittest.TestCase方法;

3.重写setUp和tearDown方法(如果有初始化条件和结束条件)。若setup()成功运行,无论测试方法是否成功,都会运行tearDown ();

4.定义测试函数,函数名以test_开头,以识别测试用例;

5.调用unittest.main()方法运行测试用例;

6.用例执行后,需要判断用例是Pass还是Fail,可以用unittest.TestCase模块的:断言

断言就是比对预期结果。如果不加断言,没有结果对比,需要手动去检查运行的结果是否符合预期。

2、使用介绍

1.要想使用unittest单元测试框架,必须得先导入:import unittest

2.查看unittest源码;

1

2

import unittest

print(help(unittest))

从打印结果中提取出unittest简易的例子:

1

2

3

4

5

6

7

8

9

10

import unittest

class IntegerArithmeticTestCase(unittest.TestCase):

    def testAdd(self):  # test method names begin with 'test'

        self.assertEqual((1 + 2), 3)

        self.assertEqual(0 + 1, 1)

    def testMultiply(self):

        self.assertEqual((0 * 10), 0)

        self.assertEqual((5 * 8), 40)

if __name__ == '__main__':

    unittest.main()

四、unittest实例

1、TestCase(测试用例)

看了官方代码后,我们自己写个例子熟悉下,并总结出规律:

1

2

3

4

5

6

7

8

9

10

11

import unittest

class TestDemo(unittest.TestCase):

    # test_sub用例

    def test_sub(self):

        self.assertEqual(2-1,1)

    # test_add用例

    def test_add(self):

        self.assertEqual(2+1,3)

if __name__ == "__main__":

    # unittest.main()是运行主函数

    unittest.main(verbosity=2)

运行结果为:

test_add (__main__.TestDemo) ... ok
test_sub (__main__.TestDemo) ... ok
----------------------------------------------------------------------
Ran 2 tests in 0.000s
OK

接下来,我们来总结一些规律:

  • 使用unittest前,需导入unittest框架。
  • TestDemo这个类必须继承unittest.TestCase,TestCase类,所有测试用例类继承的基类。
  • 类内的方法必须以test开头,比如test_add。
  • 断言:assertEqual用来断言预期结果和实际结果是否一致。当然unittest还包含很多其他断言方法,后面统一介绍。
  • 用例执行顺序。在代码中test_sub方法写在test_add前,但实际,test_add比test_sub先运行。为什么呢?unittest执行测试用例,默认是根据ASCII码的顺序加载测试用例,数字与字母的顺序为:0-9,A-Z,a-z。
  • verbosity是一个选项,表示测试结果的信息复杂度,有0、1、2 三个值。verbosity=0 : 你只能获得测试用例数总的结果;verbosity=1 (默认模式): 在每个成功的用例前面有个“.” 每个失败的用例前面有个 “F”;verbosity=2 (详细模式):测试结果会显示每个测试用例的所有相关的信息。

如下,在测试用例中写入断言:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

import unittest

#测试MathMethod类

class TestMathMethod(unittest.TestCase):

    #编写测试用例

    def test_add_two_positive(self): #测试两个正数相加

        res=MathMethod(1,1).add()

        print("1+1的结果是:",res)

        self.assertEqual(2,res,"两个正数相加出错!") #断言

    def test_add_two_zero(self): #测试两个0相加

        res = MathMethod(0, 0).add()

        print("0+0的结果是:", res)

        self.assertEqual(0, res, "两个0相加出错!") #断言

    def test_add_two_negative(self): #测试两个负数相加

        res = MathMethod(-1, -1).add()

        print("-1+(-1)的结果是:", res)

        self.assertEqual(-2, res, "两个负数相加出错!") #断言

if __name__ == '__main__':

    unittest.main()

测试用例里面的setUp函数与tearDown函数的使用:

1

2

3

4

5

6

7

8

9

10

11

class TestMathMethod(unittest.TestCase):

def setUp(self):

    print("开始准备执行测试用例!")

     

def tearDown(self):

    print("结束!")

     

#编写测试用例

def test_add_two_positive(self): #测试两个正数相加

    res=MathMethod(1,1).add()

    print("1+1的结果是:",res)

setUp函数:初始化环境(执行每条用例之前,都要执行setUp函数下面的代码,每次都要执行);

tearDown函数:清洗环境(执行每条用例之后,都要执行tearDown函数下面的代码,每次都要执行);

setUp()、tearDown()是TestCase里的方法,写在测试类中,就是方法的重写。

执行顺序是:setUp->testA->tearDown->setUp->testB>tearDown

2、TestSuit(测试集)

当测试用例全部写完,但是只想执行其中部分,可以使用TestSuit()来收集测试用例。

1

2

3

4

5

6

7

import unittest

from xxx import xxx      #测试用例的类

suite=unittest.TestSuit()

suite.addTest(测试用例的类("用例名称1"))    #用例名称用字符串的形式传入

suite.addTest(测试用例的类("用例名称2"))

suite.addTest(测试用例的类("用例名称3"))

.....

3、TestLoader(加载测试用例)

方式一:通过测试类来加载用例(loadTestsFromTestCase)

一次性加载测试用例类名1下的所有用例。

1

2

3

4

5

6

import unittest

    from xxx import xxx      #测试用例的类

    suite=unittest.TestSuit()

    loader=unittest.TestLoader()

    suite.addTest(loader.loadTestsFromTestCase(测试用例类名1)) 

    #测试用例类名直接传入

方式二:通过测试类所在的模块加载用例(loadTestsFromModule)

一次性加载测试用例模块名下的所有用例。

1

2

3

4

5

import unittest

from xxx import xxx      #测试用例模块

suite=unittest.TestSuit()

loader=unittest.TestLoader()

suite.addTest(loader.loadTestsFromTestCase(测试用例模块名))   #测试用例模块名直接传入

4、生成测试报告

方式一:使用unittest自带的TextTestRunner生成测试报告(文本格式,不推荐使用)。

TextTestRunner是一个以文本形式展示测试结果的测试运行程序类

stream 输出报告的路径,默认输出控制台;

verbosity 控制输出报告的详细程度,从0-2,越来越详细;

方式二:使用第三方模块HTMLTestRunnerNew(生成HTML格式的测试报告,推荐使用)。

file:文件

verbosity:详细程度

title:标题

description:描述

tester:作者

1

2

3

4

5

6

7

with open("接口测试报告.html","wb") as file:

    runner = HTMLTestRunnerNew.HTMLTestRunner(stream=file

                                              verbosity=2,

                                              title="接口自动化测试报告",

                                              description="接口测试V1",

                                              tester="ITester软件测试小栈")

    runner.run(suite)

方式三:使用unittest.defaultTestLoader.discover() 模糊匹配。

1

2

3

4

5

6

7

8

9

10

import unittest

import HTMLTestRunnerNew

all_testcases=unittest.defaultTestLoader.discover(contants.testcases_dir, pattern='test_*.py',top_level_dir=None)

#利用上下文管理器自动关闭资源

with open(contants.reports_html,"wb+") as file:    #选择绝对路径,把文件打开,写进内容 (报告的文件名直接写在路径里面)

    runner=HTMLTestRunnerNew.HTMLTestRunner(stream=file,

                                            title="接口自动化测试报告",

                                            description="接口测试V1",

                                            tester="ITester软件测试小栈")

    runner.run(all_testcases)

到此这篇关于Python接口自动化浅析unittest单元测试原理的文章就介绍到这了,更多相关Python接口自动化unittest单元测试内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持小编!

如果我的博客对你有帮助、如果你喜欢我的博客内容,请 “点赞” “评论” “收藏” 一键三连哦!

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

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

相关文章

二叉树的各类实现判断二叉树的递归套路

如何判断一颗二叉树是否是搜索二叉树? 搜索二叉树 每个子树头节点的左孩子比它小,右孩子比它大 经典的搜索二叉树没有重复的数 判断 将二叉树按照中序遍历,判断是否为升序 1、先将整棵树中序遍历再判断是否升序 //中序遍历public stat…

如何获取ABAP的程序事件顺序的调用堆栈

难道有激情总结下之前做过的事情,话不多说直接上图 重点在于此函数 CALL FUNCTION SYSTEM_CALLSTACK IMPORTING ET_CALLSTACK L_CSTACK_TAB. " internal table

AD9371 官方例程HDL详解之JESD204B TX侧时钟生成 (一)

AD9371 系列快速入口 AD9371ZCU102 移植到 ZCU106 : AD9371 官方例程构建及单音信号收发 ad9371_tx_jesd -->util_ad9371_xcvr接口映射: AD9371 官方例程之 tx_jesd 与 xcvr接口映射 梳理 AD9371 时钟,理解采样率和各个时钟之间的关系 …

【WCA-KELM预测】基于水循环算法优化核极限学习机回归预测研究(Matlab代码实现)

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…

详解Windows系统下面如何查看CUDA、cuDNN、Python和各个软件包的版本

文章目录 简介查看CUDA版本查看cuDNN版本查看Python版本查看Python环境中已安装软件包的版本参考 简介 这个题目网络上有很多的讲解,但是查看CUDA、cuDNN版本和查看Python与自身各个软件包是分开的,且cuDNN版本的查看方式似乎已经过时【截止2023-10-23】…

测试面试必备:HTTP请求和响应详解!

一次完整的HTTP请求过程从TCP三次握手建立连接成功后开始,客户端按照指定的格式开始向服务端发送HTTP请求,服务端接收请求后,解析HTTP请求,处理完业务逻辑,最后返回一个HTTP的响应给客户端,HTTP的响应内容同…

Windows新电脑安装环境快速运行Springboot项目

文章目录 简要步骤说明1、配置java运行环境2、配置maven环境3、下载git4、运行IDES Eclipse STS4.1 安装 lombok插件4.2 配置 maven setting.xml 地址4.3 配置 Java版本 5、顺利运行 Springboot项目 简要步骤说明 1、配置java运行环境 安装java11 2、配置maven环境 配置 setti…

【c++】遍历容器,哪一种方法速度最快?

终于有一个简单的每日一题!写完的时候甚至代码还没有编译结束!刚好借今天的每日一题探究一下一直以来的一些疑惑:容器的遍历。 题目大概是这样的: 我们一眼就看到了容器的遍历!!那么众所周知,容…

管理类联考——英语二——阅读篇——题材:经济

文章目录 2011 年,Text 2——题材:经济句意理解题-原词复现一般不选,但是要留意无中生有推断题-原词复现,注意是否“无中生有”“对象错误”“词意相反”推断题推断题主旨题-不了解阅读意思,很难做出选择 2011 年&…

R-CNN(CVPR2014)

文章目录 AbstractIntroductionObject detection with R-CNNModule designTest-time detection Visualization, ablation, and modes of errorSemantic segmentationConclusion 论文链接 源代码 Abstract 在经典PASCAL VOC数据集上测量的对象检测性能在过去几年中趋于稳定&…

MTK6737安卓核心板-MT6737核心板_4G安卓模块

MT6737核心板以竞争力的价格提供出色的性能和功耗效率,降低了BOM、GMO和内存成本,符合以价值为导向的市场需求。该核心板是一种先进的解决方案,为终端用户提供了高品质又经济实惠的体验。它与全球范围内的IMS兼容,支持VoLTE、ViLT…

【C++那些事儿】函数重载与C++中的“指针“——引用

君兮_的个人主页 即使走的再远,也勿忘启程时的初心 C/C 游戏开发 Hello,米娜桑们,这里是君兮_,我之前看过一套书叫做《明朝那些事儿》,把本来枯燥的历史讲的生动有趣。而C作为一门接近底层的语言,无疑是抽象且难度颇…

SparkSQL之LogicalPlan概述

逻辑计划阶段在整个流程中起着承前启后的作用。在此阶段,字符串形态的SQL语句转换为树结构形态的逻辑算子树,SQL中所包含的各种处理逻辑(过滤、剪裁等)和数据信息都会被整合在逻辑算子树的不同节点中。逻辑计划本质上是一种中间过…

面试时一些不能说的离职原因

“你为什么从上一家公司离职?”这个问题在面试时基本都会被问到,这是无法避免的问题。那么什么样的理由才能做到既反映实际情况,又能得到HR认可呢?以下的几种回答千万不能脱口而出。 1、毫无顾忌地说前公司的坏话 1)…

进阶JAVA篇- Map 系列集合的遍历方法与常用API

目录 1.0 Map 集合的说明 1.1 Map 集合的常用方法 1.2 Map 系列集合的特点 2.0 Map 系列集合的遍历方法(三种方法) 2.1 使用 keySet() 方法遍历 2.2 使用 entrySet() 方法遍历 2.3 使用 forEach() 方法遍历(Java 8) 1.0 Map 集合的…

数据结构与算法——选择排序法

个人简介 👀个人主页: 前端杂货铺 🙋‍♂️学习方向: 主攻前端方向,正逐渐往全干发展 📃个人状态: 研发工程师,现效力于中国工业软件事业 🚀人生格言: 积跬步…

镇魂街武神躯手游攻略,镇魂街武神躯神魂搭配

武神躯是《镇魂街》手游中令人敬畏的角色之一,技能的合理搭配至关重要。下面将详细介绍镇魂街武神躯手游攻略如何搭配武神躯的技能以取得最佳战斗效果: 关注【娱乐天梯】,获取内部福利号 1. 主动技能-刺客之心:这是武神躯的核心主…

python 桌面软件开发-matplotlib画图鼠标缩放拖动

继上一篇在 Java 中缩放拖动图片后,在python matplotlib中也来实现一个自由缩放拖动的例子: python matplotlib 中缩放,较为简单,只需要通过设置要显示的 x y坐标的显示范围即可。基于此,实现一个鼠标监听回调&#xf…

python安装的第三方库

安装第三方库的方式 先按“winR”并输入 cmd 打开命令运行窗口 输入命令: pip install 包名,比如要安装gym包就输入pip install gym 可能存在报错问题及解决办法 报错Could not fetch URL https://pypi.tuna.tsinghua.edu.cn/simple/pip/: There was …

基于Java的美食推荐管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序(小蔡coding) 代码参考数据库参考源码获取 前言 💗博主介绍:✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作者&am…