使用Google Mock可解决C++单元测试中依赖外部组件的问题。首先通过CMake等工具集成gMock,接着将代码依赖抽象为接口,如定义Database基类,再用MOCK_METHOD宏创建MockDatabase模拟其行为。在测试中,通过EXPECT_CALL设定对mock方法的调用期望,例如参数匹配、返回值和调用次数,并注入mock对象到被测类如UserService中,从而验证逻辑正确性。gMock还支持InSequence控制调用顺序、Times指定调用次数及自定义回调动作,提升测试灵活性。关键是依赖倒置与接口抽象,使测试专注逻辑而非协作者。

在C++中进行单元测试时,依赖外部组件(如数据库、网络服务或硬件接口)会使测试变得复杂且不稳定。为了解决这个问题,可以使用 mocking 技术来模拟这些依赖行为。Google Mock(简称 gMock)是 Google Test 框架的一部分,专门用于创建和管理模拟对象,帮助你隔离被测代码并验证其交互逻辑。
什么是 Mocking?
Mocking 是指用一个可控的“替身”对象替代真实依赖,这个替身能模拟真实对象的行为,并允许你检查方法是否被调用、调用次数以及参数是否正确。这在测试中非常有用,特别是当你想专注于某个类的逻辑而不受其协作者影响时。
集成 Google Mock 到项目中
Google Mock 通常与 Google Test 一起使用。你可以通过以下方式之一将其引入项目:
- 使用 CMake 和 FetchContent:适合现代 C++ 项目。
- 从源码构建并安装:下载 googletest 仓库,编译并安装到系统路径。
- 使用包管理器:如 vcpkg 或 conan 安装 gtest + gmock。
在 CMakeLists.txt 中添加:
立即学习“C++免费学习笔记(深入)”;
include(FetchContent) FetchContent_Declare( googletest URL https://github.com/google/googletest/archive/refs/tags/v1.14.0.zip ) FetchContent_MakeAvailable(googletest) <h1>启用 gmock</h1><p>target_link_libraries(your_test_target gtest_main gmock)
编写可测试代码:依赖抽象
要使用 mock,你的代码必须依赖于接口(抽象类),而不是具体实现。例如:
class Database {
public:
virtual ~Database() = default;
virtual bool save(const std::string& key, const std::string& value) = 0;
virtual std::string read(const std::string& key) = 0;
};
<p>class UserService {
public:
explicit UserService(Database* db) : db_(db) {}</p><p>bool createUser(const std::string& username) {
return db_->save("user:" + username, username);
}</p><p>private:
Database* db_;
};
这样我们就可以为 Database 创建一个 mock 实现来进行测试。
使用 Google Mock 创建模拟对象
使用 MOCk_METHOD 宏定义 mock 类的方法:
#include <gmock/gmock.h>
<p>class MockDatabase : public Database {
public:
MOCK_METHOD(bool, save, (const std::string&, const std::string&), (override));
MOCK_METHOD(std::string, read, (const std::string&), (override));
};
现在可以用这个 mock 替代真实的数据库。
在测试中使用 Mock 对象
结合 Google Test 编写测试用例:
#include <gtest/gtest.h>
#include <gmock/gmock.h>
<p>using ::testing::Return;
using ::testing::Eq;
using ::testing::ByRef;</p><p>TEST(UserServiceTest, SaveCallsDatabaseSave) {
MockDatabase mock_db;
UserService service(&mock_db);</p><p>// 设定期望:当 save 被调用时返回 true
EXPECT_CALL(mock_db, save(Eq("user:alice"), Eq("alice")))
.WillOnce(Return(true));</p><p>bool result = service.createUser("alice");
EXPECT_TRUE(result);
}
这里的关键点:
-
EXPECT_CALL设置对 mock 方法的调用期望。 - 可以匹配参数(如
Eq)、检查调用次数等。 -
WillOnce(Return(...))指定返回值。
高级特性:验证调用顺序和多次调用
Google Mock 支持更复杂的场景:
// 验证调用顺序
{
testing::InSequence seq;
EXPECT_CALL(mock_db, save("x", "y"));
EXPECT_CALL(mock_db, read("x"));
}
<p>// 匹配任意参数
EXPECT_CALL(mock_db, save).Times(2); // 被调用两次即可</p><p>// 使用回调或自定义动作
EXPECT_CALL(mock<em>db, read).WillOnce([](const std::string& key) {
return "mocked</em>" + key;
});
小结
使用 Google Mock 可以有效提升 C++ 单元测试的质量。关键是将依赖抽象成接口,在测试中用 mock 对象代替真实实现,并通过 EXPECT_CALL 验证交互行为。配合 Google Test,能够写出清晰、可靠、可维护的测试代码。
基本上就这些。只要结构合理,mock 就不会难用。










