如何在 Python 中正确模拟类定义时调用的函数(如类属性初始化中的函数)
技术百科
霞舞
发布时间:2026-01-19
浏览: 次 本文详解为何直接使用 `@patch` 无法拦截类属性中提前执行的函数调用,并提供可靠方案:结合 `patch.object` 与 `importlib.reload` 在模块重载前替换目标函数,确保类属性和实例属性均能响应 mock。
在 Python 单元测试中,当一个函数在类定义阶段(即模块导入时)被直接调用并赋值给类属性(如 class_attribute = bar()),该调用会在任何测试装饰器(如 @patch)生效前就已完成。这是因为类体代码在模块首次被 import 时即执行,而 unittest.mock.patch 的作用时机是在测试方法运行前动态替换命名空间中的对象——此时 Foo.class

要真正改变类属性的初始值,必须在类被定义之前就让 bar() 返回期望的 mock 值。核心思路是:
- 先 patch 目标函数对象本身(而非其引用路径),确保所有后续调用都受控;
- 重新加载依赖该函数的模块(如 foo.py),触发类的重新定义,使 class_attribute 重新执行 bar() 并获取 mock 返回值。
以下是可直接运行的修复示例:
# test_foo.py
import importlib
import unittest
from unittest import mock
import foo
import methods # 注意:需显式导入被 patch 的源模块
class FooTestCase(unittest.TestCase):
def test_mock_class_attribute_at_definition_time(self):
expected = "patched foo"
# 关键:patch methods 模块中的 bar 函数对象,并 reload foo 模块
with mock.patch.object(methods, "bar", return_value=expected):
importlib.reload(foo) # 重新执行 foo.py → 重新定义 Foo 类
# 此时 class_attribute 和实例属性均使用 mock 返回值
self.assertEqual(foo.Foo.class_attribute, expected)
self.assertEqual(foo.Foo().class_attribute, expected)⚠️ 注意事项:
- 必须 import methods(而非仅 from methods import bar),因为 patch.object 需要操作模块对象本身;
- importlib.reload(foo) 会重新执行 foo.py 全局代码,包括类定义,因此需确保 foo 模块无副作用(如重复注册、全局状态变更);
- 若 foo.py 依赖其他未被 reload 的模块,应一并处理其依赖关系,避免状态不一致;
- 此方案适用于单元测试隔离场景,不推荐在生产代码中频繁重载模块。
总结:类属性中“立即求值”的函数调用属于模块加载期行为,mock 必须前置到该阶段之前。patch.object + importlib.reload 是解决此类问题的标准模式,它从根源上重置了类的定义上下文,而非仅拦截运行时调用。
# 是在
# 会在
# 加载
# python
# 适用于
# 此类
# 对其
# 首次
# 而非
# 对象
# 字符串
# 命名空间
# 返回值
# Object
# 类属
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- php查询数据怎么分组_groupby分组查询配合
- Win11怎么设置组合键快捷方式_Windows1
- XML的“混合内容”是什么 怎么用DTD或XSD定
- 如何高效获取循环末次生成的 NumPy 数组最后一
- c++ namespace命名空间用法_c++避免
- Windows10如何更改桌面图标间距_Win10
- c++输入输出流 c++ cin与cout格式化输
- c++23 std::expected怎么用 c+
- Win10如何卸载WindowsDefender_
- Windows蓝屏错误0x00000018怎么处理
- 如何在Golang中处理二进制数据_Golang
- C++中的std::shared_from_thi
- Mac如何使用听写功能_Mac语音输入打字【效率技
- 如何在Golang中实现WebSocket广播_使
- Win11怎么恢复出厂设置_Win11重置此电脑保
- c# 服务器GC和工作站GC的区别和设置
- 如何使用Golang实现RPC序列化与反序列化_G
- Win11怎么查看电脑配置_Win11硬件配置详细
- 如何使用Golang构建简易投票统计功能_Gola
- Win11怎么关闭键盘按键音_Win11禁用打字声
- MySQL 中使用 IF 和 CASE 实现查询字
- Mac怎么设置鼠标滚动速度_Mac鼠标设置详细参数
- Laravel 查询 JSON 列:高效筛选包含数
- Windows11怎么自定义任务栏_Windows
- Mac如何解压zip和rar文件?(推荐免费工具)
- Go语言中正确反序列化多个同级XML元素为结构体切
- Win11如何设置系统语言_Win11系统语言切换
- c++如何实现一个高性能的环形队列(Ring Bu
- mac本地php环境如何开启curl_curl扩展
- Win11讲述人怎么关闭_Win11误触开启语音朗
- Go 中实现 Python urllib.quot
- Drupal 中 HTML 链接被重复转义导致渲染
- Python网络超时处理_健壮性设计说明【指导】
- 如何使用Golang管理跨项目依赖_Golang多
- c# Task.Yield 的作用是什么 它和Ta
- 如何解决Windows时间不准的问题?(自动同步设
- Go语言中slice追加操作的底层共享机制详解
- Windows11怎么用“记事本”自动换行与编码
- Windows如何拦截2345弹窗广告_Windo
- Windows怎样关闭Edge新标签页广告_Win
- Win10任务栏天气和资讯怎么关闭 Win10禁用
- Python文件操作优化_大文件与流处理解析【教程
- c++中的可变参数模板(variadic temp
- Win11怎么开启上帝模式_创建Windows 1
- Mac如何修复应用程序权限问题_Mac磁盘工具修复
- MAC如何隐藏文件夹及文件_MAC终端命令隐藏与第
- 如何在 Go 中高效缓存与分发网络视频流
- Win10怎样设置多显示器_Win10多显示器扩展
- Win11怎么更改账户头像_Windows 11自
- Win11怎么更改鼠标指针方案_Windows11

QQ客服