Unitest和pytest使用方法

unittest 是 Python 自带的单元测试框架,用于编写和运行可重复的测试用例。它的核心思想是通过断言(assertions)验证代码的行为是否符合预期。以下是 unittest 的基本使用方法:


1. 基本结构

1.1 创建测试类
  • 继承 unittest.TestCase,每个测试用例对应一个方法。
  • 测试方法必须 test_ 开头,否则不会被自动识别为测试用例。
import unittestclass TestMathOperations(unittest.TestCase):def test_addition(self):self.assertEqual(1 + 1, 2)  # 断言 1+1=2
1.2 前置与后置方法
  • setUp(): 在每个测试方法执行前运行(如初始化资源)。
  • tearDown(): 在每个测试方法执行后运行(如清理资源)。
  • setUpClass() / tearDownClass(): 在整个测试类的开始/结束时运行(需用 @classmethod 修饰)。
class TestExample(unittest.TestCase):@classmethoddef setUpClass(cls):print("整个测试类开始前执行")def setUp(self):print("每个测试方法开始前执行")def test_example(self):self.assertTrue(True)def tearDown(self):print("每个测试方法结束后执行")@classmethoddef tearDownClass(cls):print("整个测试类结束后执行")

2. 断言方法

unittest 提供了丰富的断言方法,常用如下:

方法说明
assertEqual(a, b)检查 a == b
assertTrue(x)检查 x 为 True
assertFalse(x)检查 x 为 False
assertRaises(Error, func, *args)检查函数 func 是否抛出 Error 异常
assertIn(a, b)检查 ab
assertIsNone(x)检查 x 是 None
def test_assertions(self):self.assertEqual(3 * 3, 9)self.assertIn(2, [1, 2, 3])with self.assertRaises(ZeroDivisionError):_ = 1 / 0

3. 运行测试

3.1 通过代码运行

在脚本末尾添加:

if __name__ == "__main__":unittest.main()
3.2 通过命令行运行
# 运行单个测试模块
python -m unittest test_module.py# 自动发现并运行所有测试(推荐)
python -m unittest discover
3.3 指定运行特定测试
# 运行单个测试类
python -m unittest test_module.TestClass# 运行单个测试方法
python -m unittest test_module.TestClass.test_method

4. 测试套件(Test Suite)

手动组织多个测试用例:

suite = unittest.TestSuite()
suite.addTest(TestMathOperations("test_addition"))
suite.addTest(TestExample("test_example"))runner = unittest.TextTestRunner()
runner.run(suite)

5. 高级用法

5.1 跳过测试

使用装饰器跳过某些测试:

@unittest.skip("跳过原因")
def test_skipped(self):self.fail("不会执行")@unittest.skipIf(condition, "条件满足时跳过")
def test_conditional_skip(self):pass
5.2 参数化测试

unittest 本身不支持参数化,但可通过第三方库(如 parameterized)实现:

from parameterized import parameterizedclass TestParameterized(unittest.TestCase):@parameterized.expand([(2, 3, 5),(0, 0, 0),])def test_add(self, a, b, expected):self.assertEqual(a + b, expected)
5.3 Mock 对象

使用 unittest.mock 模拟外部依赖:

from unittest.mock import Mockdef test_mock(self):mock_obj = Mock(return_value=42)self.assertEqual(mock_obj(), 42)

6. 示例项目结构

project/
├── my_code.py       # 被测试的代码
└── tests/├── __init__.py└── test_code.py # 测试代码

总结

unittest 是 Python 测试的基石,适合中小型项目。对于复杂场景,可以结合第三方库(如 pytest)增强功能。核心步骤:

  1. 继承 TestCase 编写测试类。
  2. 使用 test_ 前缀定义测试方法。
  3. 通过断言验证逻辑。
  4. 利用 setUp()/tearDown() 管理资源。
  5. 运行测试并分析结果。

pytest 是 Python 中最流行的第三方测试框架,以其简洁的语法、强大的功能和灵活的扩展性著称。相比 unittestpytest 更注重代码的可读性和可维护性,同时支持丰富的插件生态系统。以下是 pytest 的核心使用方法:


1. 安装 pytest

pip install pytest

2. 基本用法

2.1 编写测试函数
  • 测试函数名需以 test_ 开头(或 _test 结尾)。
  • 断言直接使用 Python 原生 assert 语句,无需记忆特定断言方法。
# test_sample.py
def test_addition():assert 1 + 1 == 2def test_list_contains():numbers = [1, 2, 3]assert 2 in numbers
2.2 运行测试
# 运行当前目录所有测试
pytest# 运行指定文件
pytest test_sample.py# 运行指定函数
pytest test_sample.py::test_addition# 显示详细输出(-v 显示用例名称,-s 打印输出)
pytest -v -s

3. 断言增强

pytest 的断言失败信息更直观,能自动展示上下文差异(如列表、字典比较):

def test_failure_example():expected = {"a": 1, "b": 2}actual = {"a": 1, "b": 3}assert expected == actual

运行后输出:

AssertionError: assert {'a': 1, 'b': 2} == {'a': 1, 'b': 3}Differing items:{'b': 2} != {'b': 3}

4. Fixture(依赖注入)

pytestfixture 机制用于管理测试的依赖资源(如数据库连接、临时文件),支持复用和共享。

4.1 定义 Fixture
import pytest@pytest.fixture
def database_connection():conn = create_db_connection()  # 初始化资源yield conn                     # 返回资源conn.close()                   # 清理资源
4.2 使用 Fixture

在测试函数中通过参数名直接调用:

def test_query(database_connection):result = database_connection.query("SELECT * FROM users")assert len(result) > 0
4.3 Fixture 作用域

通过 scope 参数控制生命周期:

@pytest.fixture(scope="module")  # 作用域:模块级(每个模块执行一次)
def shared_resource():return initialize_resource()

5. 参数化测试

使用 @pytest.mark.parametrize 对单条测试用例注入多组参数,避免重复代码。

import pytest@pytest.mark.parametrize("a, b, expected", [(2, 3, 5),(0, 0, 0),(-1, 5, 4),
])
def test_add(a, b, expected):assert a + b == expected

6. 测试异常

使用 pytest.raises 捕获并验证异常:

def test_division_by_zero():with pytest.raises(ZeroDivisionError):_ = 1 / 0

7. Mock 对象(依赖隔离)

使用 pytest-mock 插件(基于 unittest.mock)模拟外部依赖:

pip install pytest-mock

示例:

def test_mocking(mocker):mock_requests = mocker.patch("requests.get")  # 模拟 requests.getmock_requests.return_value.status_code = 200response = requests.get("https://api.example.com")assert response.status_code == 200

8. 插件扩展

pytest 支持丰富的插件,例如:

  • pytest-cov: 测试覆盖率统计
  • pytest-xdist: 并行运行测试
  • pytest-django: Django 项目集成
  • pytest-asyncio: 异步测试支持

安装插件:

pip install pytest-cov pytest-xdist

9. 项目结构

project/
├── src/                  # 源代码
│   └── my_module.py
└── tests/               # 测试代码├── __init__.py├── conftest.py      # 全局 Fixture 定义├── test_core.py└── test_api.py

10. 与 unittest 兼容

pytest 可以直接运行 unittest 风格的测试用例:

# test_unittest_style.py
import unittestclass TestOldCode(unittest.TestCase):def test_legacy(self):self.assertEqual(1 + 1, 2)

运行:

pytest test_unittest_style.py

11. 高级功能

  • 标记(Markers)
    @pytest.mark 对测试分类(如跳过、标记为慢测试):

    @pytest.mark.skip(reason="尚未实现")
    def test_unimplemented():assert False@pytest.mark.slow
    def test_long_running():# 耗时操作pass
    

    运行指定标记的测试:

    pytest -m slow    # 只运行标记为 slow 的测试
    pytest -m "not slow"  # 排除 slow 测试
    
  • Hook 函数
    自定义 pytest 行为(如修改报告输出)。


总结

pytest 的优势:

  1. 简洁性:使用原生 assert,减少样板代码。
  2. 灵活性:Fixture 机制优雅管理测试依赖。
  3. 扩展性:通过插件支持复杂场景(如异步、分布式测试)。
  4. 兼容性:无缝运行 unittestnose 测试。

适合从简单脚本到大型项目的全场景测试需求。

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

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

相关文章

git 版本提交规范

Git 提交规范(Git Commit Message Convention)是为了让项目的提交历史更加清晰、可读、便于追踪和自动化工具解析。常见的规范之一是 Conventional Commits,下面是一个推荐的格式规范: 🌟 提交信息格式(Con…

stat判断路径

int stat(const char *pathname, struct stat *buf); pathname:用于指定一个需要查看属性的文件路径。 buf:struct stat 类型指针,用于指向一个 struct stat 结构体变量。调用 stat 函数的时候需要传入一个 struct stat 变量的指针&#xff0…

学习Docker遇到的问题

目录 1、拉取hello-world镜像报错 1. 检查网络连接 排查: 2. 配置 Docker 镜像加速器(推荐) 具体解决步骤: 1.在服务器上创建并修改配置文件,添加Docker镜像加速器地址: 2. 重启Docker 3. 拉取hello-world镜像 2、删除镜像出现异常 3、 容器内部不能运行ping命令 …

安宝特案例 | AR如何大幅提升IC封装厂检测效率?

前言:如何提升IC封装厂检测效率? 在现代电子产品的制造过程中,IC封装作为核心环节,涉及到复杂处理流程和严格质量检测。这是一家专注于IC封装的厂商,负责将来自IC制造商的晶圆进行保护、散热和导通处理。整个制程繁琐…

【Linux网络与网络编程】07.应用层协议HTTPS

HTTP 协议内容都是按照文本的方式明文传输的,这就导致在传输过程中出现一些被篡改的情况。HTTPS 就是在 HTTP 协议的基础上引入了一个加密层的应用层协议。 1. 基础概念 1.1 加密与解密 加密就是把明文(要传输的信息)进行一系列变换&#x…

【k8s】PV,PVC的回收策略——return、recycle、delete

PV 和 PVC 的回收策略主要用于管理存储资源的生命周期,特别是当 PVC 被删除时,PV 的处理方式。回收策略决定了 PV 在 PVC 被删除后的行为。 回收策略的类型 Kubernetes 提供了三种主要的回收策略,用于管理 PV 的生命周期: Reta…

2023蓝帽杯初赛内存取证-2

直接使用mimikatz插件来获取用户密码: vol.py --plugin/opt/volatility/plugins -f memdump.mem --profile Win7SP1x64 mimikatz 答案:3w.qax.com

使用dompurify修复XSS跨站脚本缺陷

1. 问题描述 漏洞扫描说有一个低危漏洞,容易被跨站脚本攻击XSS。 2. 使用dompurify修复 DOMPurify is a DOM-only, super-fast, uber-tolerant XSS sanitizer for HTML, MathML and SVG. 简单来说,我们可以使用 dompurify 处理xss跨站脚本攻击。 2.…

【c语言】指针和数组笔试题解析

一维数组: //数组名a如果既不单独放在sizeof()中,也不与&结合,那么就表示数组首元素的大小 //a一般表示数组首元素地址,只有两种情况表示整个数组,sizeof(arr)表示整个数组的大小,&arr表示数组的地址 int a[]…

机器人进阶---视觉算法(六)傅里叶变换在图像处理中怎么用

傅里叶变换在图像处理中怎么用 傅里叶变换的基本原理应用场景Python代码示例逐行解释总结傅里叶变换在图像处理中是一种重要的工具,它将图像从空间域转换到频域,从而可以对图像的频率特性进行分析和处理。傅里叶变换在图像滤波、图像增强、图像压缩和图像分析等方面都有广泛应…

深度学习与总结JVM专辑(七):垃圾回收器—CMS(图文+代码)

CMS垃圾收集器深度解析教程 1. 前言:为什么需要CMS?2. CMS 工作原理:一场与时间的赛跑2.1. 初始标记(Initial Mark)2.2. 并发标记(Concurrent Mark)2.3. 重新标记(Remark&#xff09…

数据采集:AI 发展的基石与驱动力

人工智能(AI)无疑是最具变革性的技术力量之一,正以惊人的速度重塑着各行各业的格局。从智能语音助手到自动驾驶汽车,从精准的医疗诊断到个性化的推荐系统,AI 的广泛应用已深刻融入人们的日常生活与工作的各个层面。而在…

从信息泄露到内网控制

0x01 背景 之前常见用rce、文件上传等漏洞获取webshell,偶然遇到一次敏感信息泄露获取权限的渗透,简单记录一下过程。 0x02 信息泄露 发现系统某端口部署了minio服务,经过探测发现存在minio存储桶遍历 使用利用工具把泄露的文件全部整理一…

《门》凡是过往,皆为序曲。我们的爱,和最初一样

《门》凡是过往,皆为序曲。我们的爱,和最初一样 夏目漱石,本名夏目金之助,笔名漱石,日本近代作家,代表作有《三四郎》《门》《从此以后》《我是猫》《心》《明暗》等。 竺家荣 译 文章目录 《门》凡是过往&…

衡石ChatBI:依托开放架构构建技术驱动的差异化数据服务

在当今数字化浪潮中,企业对数据价值的挖掘和利用需求日益增长。BI(商业智能)工具作为企业获取数据洞察的关键手段,其技术架构的创新与发展至关重要。衡石科技的Chat BI凭借其独特的开放架构,在BI领域脱颖而出&#xff…

oracle中错误总结

oracle中给表起别名不能用as,用as报错 在 Oracle 数据库中,​​WITH 子句(即 CTE,公共表表达式)允许后续定义的子查询引用前面已经定义的 CTE​​,但 ​​前面的 CTE 无法引用后面的 CTE​​。这种设计类似…

NLP高频面试题(五十)——大模型(LLMs)分词(Tokenizer)详解

在自然语言处理(NLP)任务中,将文本转换为模型可处理的数字序列是必不可少的一步。这一步通常称为分词(tokenization),即把原始文本拆分成一个个词元(token)。对于**大型语言模型(LLM,Large Language Model,大型语言模型)**而言,选择合适的分词方案至关重要:分词的…

优化WAV音频文件

优化 WAV 音频文件通常涉及 减小文件体积、提升音质 或 适配特定用途(如流媒体、广播等)。以下是分场景的优化方法,涵盖工具和操作步骤: 一、减小文件体积(无损/有损压缩) 1. 无损压缩 转换格式&#xff1…

ORACLE SQL输入的变量由于隐式转换无法使用索引的分析优化

近期,某客户在巡检分析AWR报告时,发现有个TOP SQL的执行效率偏慢,检查分析SQL,发现数据块读取量高,分析执行计划,发现有个查询条件未使用到索引; 对执行计划及表上的字段、索引进行分析&#x…

【锂电池SOH估计】RF随机森林锂电池健康状态估计,锂电池SOH估计(Matlab完整源码和数据)

目录 效果一览程序获取程序内容代码分享研究内容基于随机森林(RF)的锂电池健康状态(SOH)估计算法研究摘要1. 引言2. 锂电池SOH评估框架3. 实验与结果分析4. 未来研究方向6. 结论效果一览 程序获取 获取方式一:文章顶部资源处直接下载:【锂电池SOH估计】RF随机森林锂电池…