0

0

如何使用Google Mock为c++接口创建测试桩(mock)? (单元测试进阶)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-22 16:24:37

|

323人浏览过

|

来源于php中文网

原创

Google Mock 仅支持对含纯虚函数的抽象类进行模拟,要求接口类声明虚析构函数、所有待mock函数为virtual且签名一致,使用MOCK_METHOD宏声明,EXPECT_CALL须在调用前设置并按序匹配。

如何使用google mock为c++接口创建测试桩(mock)? (单元测试进阶)

Google Mock 能为 C++ 接口生成可控制行为的测试桩,但前提是接口必须是**抽象类(含纯虚函数)**,且测试类需继承 MockClass 并用 EXPECT_CALL 配置调用预期。直接对普通类、非虚函数或 final 类 mock 会编译失败或行为不可控。

定义接口时必须使用纯虚函数

Google Mock 只能 mock 继承体系中的虚函数。如果接口类没有 virtual 关键字,或者析构函数不是 virtual,mock 类无法正确接管调用,可能导致未定义行为或内存泄漏。

  • 接口类必须声明 virtual ~InterfaceName() = default;(推荐 default,避免空实现遗漏)
  • 所有待 mock 的成员函数必须是 virtual,且返回类型、参数列表、const 修饰需完全一致
  • 避免在接口中定义非虚函数——它们无法被重写,mock 对象调用时将执行基类默认实现(如果有),而非你期望的模拟逻辑

使用 MOCK_METHOD 声明 mock 函数

MOCK_METHOD 是 Google Mock 1.10+ 推荐的宏,替代旧版 MOCK_FUNCTION 或手写 ON_CALL。它自动处理 const 性、ref-qualifiers 和 noexcept,减少拼写错误。

class MockDownloader : public DownloaderInterface {
 public:
  MOCK_METHOD(bool, Download, (const std::string& url, std::string* out), (override));
  MOCK_METHOD(int, GetTimeoutMs, (), (const, override));
};
  • 括号内三组内容依次为:返回类型、函数名、参数列表(含括号)、属性说明(如 overrideconst
  • 参数列表必须和原接口完全一致,包括 std::string*const std::string&区别
  • 漏写 override 不报错但失去编译期检查;漏写 const 会导致 const 成员函数调用匹配失败

EXPECT_CALL 必须在调用前设置,且匹配顺序敏感

EXPECT_CALL 不是“声明”,而是“设定某次(或某几轮)调用的预期行为”。它只对后续发生的调用生效,且 Google Mock 按注册顺序尝试匹配——先注册的规则优先匹配。

飞书知识问答
飞书知识问答

飞书平台推出的AI知识库管理和智能搜索工具

下载

立即学习C++免费学习笔记(深入)”;

  • TEST 函数里、被测代码执行前调用 EXPECT_CALL,否则调用发生后才设预期,会触发 "Uninteresting mock function call" 警告或测试失败
  • 同一 mock 对象上多次 EXPECT_CALL 同一函数时,按书写顺序匹配:第一次调用走第一条,第二次走第二条,依此类推
  • .Times(2) 表示允许两次调用,但若实际只发生一次,测试仍失败;用 .Times(AtLeast(1)) 更适合验证“至少发生”场景
  • 不加 .WillOnce().WillRepeatedly() 时,默认返回类型的默认值(如 bool 返回 false,指针返回 nullptr

链接时需同时链接 gmockgtest

#include 不足以通过链接。Google Mock 的实现分布在独立源文件中,CMake 或 Makefile 中漏掉库依赖会导致 undefined reference to testing::internal::MockRepository::GetRepository() 等符号错误。

  • CMake 中确保 target_link_libraries(your_test gtest gmock),且 gmockgtest 之后(因 gmock 依赖 gtest)
  • 若使用 pkg-config,应同时包含 gmockgtest--libs 输出,不能只用其中一个
  • 注意:Google Mock 1.10+ 默认启用 abseil 支持,若项目禁用 abseil,需定义 GTEST_NO_ABSL=1 并确保头文件路径不含 abseil

最常被忽略的是虚析构函数和 EXPECT_CALL 的时机——前者导致运行时崩溃难以定位,后者让测试看似通过实则没验证任何行为。mock 不是魔法,它是基于 C++ 多态机制的显式契约,每处 virtualoverrideEXPECT_CALL 都必须精准对齐接口定义。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

338

2023.08.02

java多态详细介绍
java多态详细介绍

本专题整合了java多态相关内容,阅读专题下面的文章了解更多详细内容。

15

2025.11.27

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

526

2023.09.20

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

526

2023.09.20

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1050

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

86

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

457

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

11

2026.01.19

Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

9

2026.01.22

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号