1、引言

我有一个朋友是做Python自动化测试的。前几天他告诉我去参加一个大厂面试被刷了。

我问他是有没有总结被刷下来的原因。他说面试官问了一些 pytest 单元测试框架相关的知识,包括什么插件系统和用力筛选。但是他所在的公司用的技术是基于 unittest 的,没有用过 pytest。

我跟他说你可以和技术面试官说明,在实际过程当中你没有使用过 pytest,但是你可以后面再学。这哥们说:我就是这样跟面试官说的,但是面试官告诉我 pytest 现在已经是行业里面的主流,还在坚持用 unittest 说明我的技术已经过时了,没有跟上现在测试领域的发展。

实际上他在面试之前已经了解过 pytest 的一些基础用法,但是网上的一些资料,都是停留在用法和一些知识点的讲解,没有深入到 pytest 内部运行和一些高级特性。所以被问到的时候,自己临时抱佛脚的一些知识都没有用上。

后面我给这位朋友补习了一些关于Python的高级特性。现在我连同基础部分的内容一起贴出来,希望对Python自动化测试的一些朋友有所帮助。

2、为什么用单元测试框架?

首先我要说明一下什么是单元测试框架?

unittest 和 pytest 都是单元测试框架。单元测试指的是在编程过程当中形成的对函数或者是类下面的方法进行测试的一个过程。

在不使用任何框架的前提下,其实也是可以进行单元测试的。比如我们可以通过 if 判断 、异常处理或者是其他的流程控制来表示测试是否通过。

def add(a, b):
    return a + b
    
def test_add():
    ret = add(3, 4)
    if ret == 7:
        print("add 函数的测试通过")
    else:
        print("add 函数的测试失败")

如果要运行这个用例,需要手工调用 test_add 这个函数:

test_add()

接下来,使用 python 运行这个文件,就能得到测试结果:

python test_add.py

虽然说上面我们通过 if 判断,对一个函数进行了测试,而且得到了测试结果,但是流程是比较复杂的:

  • 首先我们需要人工去判断结果,
  • 第2我们需要通过 Python去运行模块。
  • 第3,我们还需要显性的去调用 test_add 这个函数。

这还只是在我们只测试了一个函数的情况下,当需要测试的函数或者类越来越多的时候,这个过程会越来越复杂。

而使用单元测试框架,可以极大的简化我们对单元测试的过程,使用单元测试框架以后,框架会帮我们自动去收集用例、运行用例、生成报告。

3、pytest 的基础使用

上面的测试代码使用 pytest 编写,可以这样写。

def add(a, b):
    return a + b
    
def test_add():
    assert 7 == add(3,4)

写完测试用例以后,我们只需要在目录下输入pytest 指令,就可以自动运行用例,而且呢结果会直接显示在命令行的下方。

上面讲的是单元测试过程,也就是说对某个函数或者是类下面的方法进行测试,有的人可能会不理解。在实际工作过程当中很少进行单元测试啊,测试人员做的更多的是接口测试,UI测试,pytest 怎么用呢?

实际上不管是接口测试还是UI测试,都是可以使用 pytest。当你进行接口测试的时候,你只需要把访问接口的过程封装成一个Python函数。

def visit_api():
    print("访问接口,得到结果...")
    return response
    
def test_api():
    assert expected_response == visit_api()

当你进行 web测试的时候,你只需要操作浏览器的过程封装成一个函数?

def browser_method():
    print("点点点")
    return ui_response
    
def test_web():
    assert expected_response == browser_method()

在这种情况下。接口访问和web操作都是以函数形式存在的,我们直接去测试这个 Python 函数,也是一个单元测试的过程。

因为 pytest 是一个第三方的框架,所以我们先要安装。安装方式非常简单,只需要通过 pip 这个包管理工具安装就可以了。

pip install -U pytest

安装完成以后,我们可以向使用上面的那个例子一样:

  • 第1步:定义一个测试函数,这个测试函数通常会调用被测函数。
  • 第2步:使用assert断言,assert 后面可以跟任意的 Python 条件表达式。
assert 4 < 5
assert "yuze" in "yuze wang"
assert isinstance(6, int)

测试用例函数有 2 个注意事项:

  • 函数名称以test_开头;
  • 保存测试用例的文件以test_*.py 的形式或者 *_test.py 的形式。

例行用了以后呢,在命令行当中会显示4个部分的内容:

  • 第1个部分,测试用例和通过的结果,
  • 第2个部分,失败用例回溯信息,
  • 第3个部分,输出捕获信息,
  • 第4个部分,总结信息。

在拍test当中通过的测试用例,不会有特别详细的结果,但是这是失败的测试用例默认会有非常详细的结果,而且会帮你捕获错误原因。

4、测试夹具(Fixture)是什么?

在测试过程当中,有时你需要提前给你的测试用例去准备一个运行环境。这个测试环境通常来说被称为测试夹具(Fixture),又被称为固定装置、测试固件等。

  • 当你要测试一个电器的时候,你需要提供不同的输入电压电流的环境,
  • 当你测试一台电脑网络的时候,必须要提供网络环境,
  • 当你要测试一个手机游戏能否被安装时,你需要提供一台手机环境,
  • 当你要测试一个软件能否登录的时候,你需要准备用户名和密码这样的用户环境,
  • 当你要测试一个数据库能否操作的时候,需要提供数据库的连接环境。

现在我们来举一个夹具的例子,我们需要测试一个注册的函数。这个注册函数提供两个参数,第1个参数是手机号,第2个参数是密码。注册函数的逻辑就是对手机号码和密码进行校验,如果通过校验表示注册成功,如果不通过表示注册失败。

def is_mobile(number: str):
    if re.search(r"^1[3-9][0-9]{9}$", number):
        return True
    return False

def register(mobile, password):
    if is_mobile(mobile) and len(password) >= 6:
        return "success"
    return "fail"

def test_register():
    assert "success" == register("13177778888", "123456")

这个测试用例并没有什么问题,但是它存在优化的空间。一个优化的空间是每个手机号码都是我们手工生成的,当需要编写多组数据测试时,会有一点费时间。现在我们可以编写一个函数,自动生成一个手机号码,当我有多组数据需要测试的时候,我只需要重复调用生成手机号码的函数,就可以获取测试数据。这个生成手机号码的函数呢,就是一个夹具。

def gen_a_mobile():
    """随机生成 13 开头的手机号码。"""
    random_num = "".join([str(random.randint(1,9)) for i in range(9)])
    return "".join(["13", random_num])

pytest 提供了一种叫做依赖注入的机制,当一个函数被声明为夹具的时候,可以在测试函数中传入这个夹具的名称,pytest会自动调用它。

import random
import pytest

@pytest.fixture
def gen_a_mobile():
    """随机生成 13 开头的手机号码。"""
    random_num = "".join([str(random.randint(1,9)) for i in range(9)])
    return "".join(["13", random_num])

def test_register(gen_a_mobile):
    assert "success" == register(gen_a_mobile, "123456")

pytest 当中的夹具系统非常非常的灵活,后面如果有时间的我专门写文章跟大家讲解夹具系统。

5、数据驱动和参数化

现在我们编写的函数和测试用例是1对1的关系,也就是说,当你想测试某个功能场景的时候,你必须要去编写一个对应的测试函数。当测试的场景越来越多,测试数据越来越复杂的情况下,需要编写更多的测心率函数,而这些函数的逻辑基本上是重复的。

在 pytest 当中可以使用参数化这种测试手段,简化编写用例函数的过程。我们并不需要为每一组测试数据单独去编写一个测试函数,而是采取多种数据共用一个函数的方式。如果测试操作几乎一致,可以重复使用这一个函数进行测试。

import pytest

cases = [
    (1, 2, 3),
    ("hello", "world", "hello world"),
    (1, "world", "1world")
]

@pytest.mark.parametrize("a,b,expected", cases)
def test_add(a, b, expected):
    assert expected == add(a, b)

在这个例子当中,cases这个变量存储了三组测试用例的数据,每一组测试数据用一个元组表示,元组的第1个元素代表a,第2个元素代表B,第3个元素代表预期结果。

@pytest.mark.parametrize("a,b,expected", cases) 这一行代码的意思是说,每一次从cases变量当中取出一组测试数据。分别用a、b 、expected 三个变量接收,然后我们把这三个变量名作为函数的参数传递到 test_add 当中,就实现了参数化的过程。

当使用这一种参数化的手段进行测试的时候,测试数据和测试函数是多对一的关系,对于多组测试数据,我们只需要去编写一个测试函数,极大的提升了代码编写效率。

6、测试报告和插件

最后我们来说一下测试报告。pytest 当中的测试报告,通常是以插件的形式生成的,如果你想生成一个html格式的测试报告,可以先安装 pytest-html 这个插件。

pip install pytest-html

接下来你需要在运行用例的时候,在 pytest 命令后面加上 --html=<测试报告名称>.html

pytest --html=report.html

当运行完用例以后,你可以在当前目录下找到一个 report.html 的文件,打开就可以查看测试报告了。

pytest 之所以成为主流,有很多的原因,其中最重要的一个原因是因为其强大的插件系统。任何一个程序员,只要遵循一些简单的规范,就可以轻易的编写插件。后面我们再跟大家深入去研究 pytest 当中的夹具系统,插件系统和钩子函数这些特性。

技术面试没过,居然是没有用pytest测试框架的更多相关文章

  1. pytest测试框架 -- 简介

    一.pytest测试框架简介: (1)pytest是python的第三方测试框架,是基于unittest的扩展框架,比unittest更简洁,更高效. (2)pytest框架可以兼容unittest用 ...

  2. Pytest测试框架(一):pytest安装及用例执行

    PyTest是基于Python的开源测试框架,语法简单易用,有大量的插件,功能非常多.自动检测测试用例,支持参数化,跳过特定用例,失败重试等功能. 安装 pip install -U pytest  ...

  3. Pytest测试框架(二):pytest 的setup/teardown方法

    PyTest支持xUnit style 结构, setup() 和 teardown() 方法用于初始化和清理测试环境,可以保证测试用例的独立性.pytest的setup/teardown方法包括:模 ...

  4. Pytest测试框架(三):pytest fixture 用法

    xUnit style 结构的 fixture用于初始化测试函数, pytest fixture是对传统的 xUnit 架构的setup/teardown功能的改进.pytest fixture为测试 ...

  5. Pytest测试框架(五):pytest + allure生成测试报告

    Allure 是一款轻量级.支持多语言的开源自动化测试报告生成框架,由Java语言开发,可以集成到 Jenkins. pytest 测试框架支持Allure 报告生成. pytest也可以生成juni ...

  6. 『德不孤』Pytest框架 — 1、Pytest测试框架介绍

    目录 1.什么是单元测试框架 2.单元测试框架主要做什么 3.单元测试框架和自动化测试框架有什么关系 4.Pytest测试框架说明 5.Pytest框架和Unittest框架区别 (1)Unittes ...

  7. Pytest测试框架入门到精通(一)

    Python测试框架之前一直用的是unittest+HTMLTestRunner,听到有人说Pytest很好用,所以这边给大家介绍一下Pytest的使用 pytest是一个非常成熟的全功能的Pytho ...

  8. Pytest测试框架一键动态切换环境思路及方案

    前言 在上一篇文章<Pytest fixture及conftest详解>中,我们介绍了fixture的一些关键特性.用法.作用域.参数等,本篇文章将结合fixture及conftest实现 ...

  9. python pytest测试框架介绍二

    在介绍一中简单介绍了pytest的安装和简单使用,接下来我们就要实际了解pytest了 一.pytest的用例发现规则 pytest可以在不同的函数.包中发现用例,发现的规则如下 文件名以test_开 ...

随机推荐

  1. 第11.6节 Python正则表达式的字符串开头匹配模式及元字符“^”(插入符、脱字符)功能介绍

    符号"^"为插入符,也称为脱字符,在Python中脱字符表示匹配字符串的开头,即字符串的开头满足匹配模式的要求.这个功能有点类似搜索函数match,只是这是通过搜索模式来指定,而m ...

  2. Model/View开发小结

    老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 Model/View开发是PyQt和Qt中重要的框架之一,老猿认为另外两个就是信号槽机制和事件机制, ...

  3. python 读取目录下的文件

    参考方法: import os path = r'C:\Users\Administrator\Desktop\file' for filename in os.listdir(path): prin ...

  4. 单次期望 O(1) 的RMQ

    膜万弘,太强了!!! 刚刚变态的zjjws想要将一个需要 \(RMQ\) 问题的时间和空间都卡成 \(O(n)\) ,就在可怜的蒟蒻 Point_King 一筹莫展之时万弘他出现了,给予了本蒟蒻光明和 ...

  5. 【题解】NOI 系列题解总集

    每次做一道 NOI 系列的估计都很激动吧,对于我这种萌新来说( P1731 [NOI1999]生日蛋糕 练习剪枝技巧,关于剪枝,欢迎看我的垃圾无意义笔记 这道题是有一定难度的,需要运用各种高科技剪枝( ...

  6. "利用python进行数据分析"学习记录01

    "利用python进行数据分析"学习记录 --day01 08/02 与书相关的资料在 http://github.com/wesm/pydata-book pandas 的2名字 ...

  7. 【学习笔记】使用 bitset 求解较高维偏序问题

    求解五维偏序 给定 \(n(\le 3\times 10^4)\) 个五元组,对于每个五元组 \((a_i, b_i, c_i, d_i, e_i)\),求存在多少个 \(1\le j\le n\) ...

  8. 从隔壁兄弟那爬点网页数据,整理成word文档

    package com.parse_report; import com.report.utils.ParseUrl; /** * Create by JIUN 2018/7/14 */ public ...

  9. PhotoSwipe用法

    1.自动识别data-size问题,添加以下代码 gallery = new PhotoSwipe(pswpElement, PhotoSwipeUI_Default, items, options) ...

  10. django 验证码

    1.django 缓存设置 django的六种缓存(mysql+redis) :https://www.cnblogs.com/xiaonq/p/7978402.html#i6 1.1 安装Djang ...