
本教程深入探讨了在python单元测试中使用`unittest.mock`模拟类方法抛出异常时,`call_count`意外为零的常见困惑。文章将阐明`patch`类时,方法调用计数应针对模拟的实例对象而非模拟类本身,并通过详尽的代码示例和解释,指导开发者正确地设置`side_effect`并断言方法调用,确保测试逻辑的准确性。
在编写单元测试时,我们经常需要模拟(mock)外部依赖的行为,包括模拟这些依赖抛出异常的情况。unittest.mock库是Python中实现这一目标的强大工具。然而,在使用patch来模拟一个类及其方法,并期望该方法抛出异常时,开发者可能会遇到一个令人困惑的问题:即使方法确实被调用并成功抛出异常(且异常可能被捕获),其call_count却显示为0。本文将深入分析这一现象的根本原因,并提供正确的解决方案。
考虑一个服务类UploadService,其中包含一个upload方法,该方法内部调用了Blob类的一个实例方法upload_from_string。我们希望测试当upload_from_string方法抛出异常时,UploadService的异常处理逻辑是否正确。
以下是相关的代码示例:
upload_service.py
import json
import logging
# 假设这些是外部库的类和异常
class GoogleCloudError(Exception):
pass
class Blob:
def __init__(self, name, bucket):
self.name = name
self.bucket = bucket
def upload_from_string(self, data, content_type):
"""模拟上传文件到云存储"""
# 实际实现可能包含与云服务交互的逻辑
print(f"Uploading {self.name} to {self.bucket} with data: {data}")
# 这里为了模拟,不实际上传
class UploadService:
def __init__(self, bucket_name):
self.bucket_name = bucket_name
def upload(self, name, data):
try:
# 实例化 Blob 对象
gcs_blob = Blob(name, self.bucket_name)
# 调用实例方法
gcs_blob.upload_from_string(data=json.dumps(data), content_type="application/json")
return "Upload successful"
except GoogleCloudError as e:
logging.exception(f"Error uploading file '{name}' to '{self.bucket_name}'")
return f"Upload failed: {e}"
test_upload_service.py (原始的、有问题的测试)
import unittest
from unittest.mock import patch, Mock
from upload_service import UploadService, Blob, GoogleCloudError
class TestUploadService(unittest.TestCase):
def test_upload_failure(self):
us = UploadService("my-test-bucket")
test_data = {"key": "value"}
test_name = "test-file.json"
with patch("upload_service.Blob") as MockedBlobClass:
# 获取模拟的 Blob 实例
gcs_blob_instance = MockedBlobClass.return_value
# 设置实例方法在调用时抛出异常
gcs_blob_instance.upload_from_string.side_effect = GoogleCloudError("Google Cloud upload failed")
# 调用待测试的方法
result = us.upload(test_name, test_data)
# 断言异常被处理
self.assertIn("Upload failed", result)
# 错误的断言:尝试在 MockedBlobClass 上检查 call_count
# 预期这里是1,但实际是0
self.assertEqual(1, MockedBlobClass.upload_from_string.call_count)
运行上述测试,会得到如下错误:
AssertionError: 1 != 0 Expected :1 Actual :0
这表明尽管side_effect被正确触发,并且异常被捕获,但MockedBlobClass.upload_from_string.call_count却为0。
问题的核心在于对unittest.mock.patch如何模拟类的理解。当使用patch("module.ClassName")时,MockedClassName实际上是一个模拟的类对象。这意味着:
AiTxt 文案助手
AiTxt 利用 Ai 帮助你生成您想要的一切文案,提升你的工作效率。
98
查看详情
kedClassName.return_value才是SUT中实际操作的“实例”。因此,MockedBlobClass.upload_from_string实际上是一个从未被调用的Mock对象,因为它代表的是“类方法”或“未实例化的类上的方法”。而真正被调用的是MockedBlobClass.return_value.upload_from_string。
要解决这个问题,我们需要将call_count的断言指向正确的Mock对象,即模拟的实例对象的方法。在我们的测试代码中,gcs_blob_instance就是这个模拟的实例对象。
以下是修正后的测试代码:
test_upload_service.py (修正后的测试)
import unittest
from unittest.mock import patch, Mock
from upload_service import UploadService, Blob, GoogleCloudError
class TestUploadService(unittest.TestCase):
def test_upload_failure_corrected(self):
us = UploadService("my-test-bucket")
test_data = {"key": "value"}
test_name = "test-file.json"
with patch("upload_service.Blob") as MockedBlobClass:
# 获取模拟的 Blob 实例
gcs_blob_instance = MockedBlobClass.return_value
# 设置实例方法在调用时抛出异常
gcs_blob_instance.upload_from_string.side_effect = GoogleCloudError("Google Cloud upload failed")
# 调用待测试的方法
result = us.upload(test_name, test_data)
# 断言异常被处理
self.assertIn("Upload failed", result)
# 正确的断言:在模拟的实例方法上检查 call_count
self.assertEqual(1, gcs_blob_instance.upload_from_string.call_count)
# 也可以通过 MockedBlobClass().upload_from_string 来访问,效果相同
# self.assertEqual(1, MockedBlobClass().upload_from_string.call_count)
通过将断言从MockedBlobClass.upload_from_string.call_count改为gcs_blob_instance.upload_from_string.call_count,测试将成功通过。这是因为gcs_blob_instance正是UploadService.upload方法中实际操作的Blob实例的模拟。
在Python单元测试中使用unittest.mock.patch模拟类及其方法时,正确理解模拟类和模拟实例之间的区别至关重要。当被测试代码实例化一个类并调用其方法时,方法调用实际上发生在模拟的实例对象上。因此,设置side_effect和断言call_count都应针对这个模拟实例的方法。遵循这些原则,可以避免常见的call_count为零的困惑,并编写出更准确、更可靠的单元测试。
以上就是解决Python单元测试中Mock异常方法调用计数为零的问题的详细内容,更多请关注其它相关文章!
# 测试中
# seo内容收集技术
# 无极seo优化
# 天水 网站建设招聘
# 姑苏网站建设哪里有
# 网站推广宝有用吗
# 东城抖音seo策划公司
# 网推网站建设
# 贵州短视频seo价格
# 关键词排名有什么优点
# 长治网站建设企业
# 多线程
# 如何处理
# 如何使用
# 数据处理
# 是在
# python
# 为零
# 是一个
# 的是
# 抛出
# 代码可读性
# 区别
# 云存储
# google
# ai
# 工具
# 云服务
# app
# go
# json
# js
相关文章:
Django表单提交验证失败后保持字段值不刷新
MinIO大规模对象列表性能瓶颈深度解析与外部元数据管理策略
Windows10怎么开启夜间模式 Windows10系统设置调整色温与亮度缓解夜间用眼疲劳【教程】
邮编格式怎么匹配地址_根据邮编格式快速匹配详细地址的技巧
Lar*el的路由模型绑定怎么用_Lar*el Route Model Binding简化控制器逻辑
b站怎么删除评论_b站评论管理与删除操作
Odoo 16:在表单视图中基于当前记录动态修改Tree视图属性
Golang如何实现微服务鉴权与权限控制_Golang微服务鉴权与权限管理实践
谷歌邮箱注册显示错误Gmail服务器异常与延迟处理
sublime如何只显示或隐藏特定类型文件_sublime侧边栏文件过滤
夸克浏览器网页版最新地址 夸克浏览器官方入口合集
c++ 获取系统当前时间 c++时间戳获取方法
微信网页版登录教程_微信网页版登录入口在哪
如何提高微信支付的安全性_微信支付安全防护与设置建议
SteamMachine定价或为699美元 大家想入手吗?
漫蛙2正版漫画站 漫蛙2网页版快速访问入口
mc.js游戏直达 mc.js网页免下载版本秒进地址
Win11 BitLocker密码忘了怎么办 Win11找回BitLocker恢复密钥方法【解决】
迅雷下载到U盘速度很慢怎么办_迅雷U盘下载慢优化方法
AO3最新镜像入口 Archive of Our Own官方平台访问
苹果手机指南针不准怎么校准 传感器校准方法详解【建议收藏】
PHP教程:将数据库查询结果动态展示到HTML Textarea的最佳实践
处理嵌套交互式控件:前端可访问性指南
2026年发布! 美少女养成动作RPG《神剑少女战记》发布实机演示
使用J*aScript检测输入元素是否包含在特定类中
腾讯视频怎么举报不良内容_腾讯视频内容举报流程与违规信息处理方法
TikTok国际版官网直达_TikTok国际版官网直达进入在线观看
excel怎么制作工资条 excel快速生成工资条的方法
整合Supabase认证与Django模型:跨模式迁移的解决方案
Go语言中高效处理x-www-form-urlencoded表单数据
win11怎么清理更新缓存 Win11删除Windows Update下载文件释放空间【技巧】
EMS快递官网app_中国邮政速递物流手机客户端
Go Martini框架:动态服务解码后的图片内容
Yandex搜索引擎官网入口_俄罗斯Yandex免登录一键直达
Tabulator表格中精确实现日期时间排序的指南
GemBox Document HTML转PDF垂直文本渲染问题及解决方案
Node.js CSV 数据处理:基于字段值条件过滤整条记录的策略
J*aScript DOM操作:高效清空列表元素的策略与实践
Yandex官网搜索引擎免登录_俄罗斯Yandex一键直达入口
如何更改在 Excel 中打开超链接时的默认浏览器
cad怎么合并重叠的线段_cad清理重复重叠线条的操作方法
在Runstone环境中高效处理TasteDive API的JSON数据
c++如何使用Meson构建系统_c++比CMake更快的构建工具
word邮件合并后日期格式不对怎么改_Word邮件合并日期格式修改方法
如何在低配置电脑上搭建轻量级J*a环境_占用更小的环境选择技巧
QQ邮箱电脑版登录入口_QQ邮箱官方网站登录平台
Win10怎么设置静态IP地址 Win10手动配置IP地址步骤【指南】
J*aScript生成器_j*ascript异步迭代
html怎么运行外部js文件中的函数_运html外js文件函数法【技巧】
Yandex官网免登录入口_俄罗斯Yandex搜索引擎一键访问
*请认真填写需求信息,我们会在24小时内与您取得联系。