【问题标题】:C++ unitTest with LinkTime substitution使用 LinkTime 替换的 C++ 单元测试
【发布时间】:2020-02-17 09:53:41
【问题描述】:

我正在阅读 James W. Grenning 的“嵌入式 C 的测试驱动开发”。

我想使用 Visual Studio Community 2019 和 gTest 重现带有“链接时间替换”的案例。

我有以下代码:

production_code 静态链接库

foo.cpp

#include "foo.h"

int foo(int x) {
    return x + 1;
}

foo.h

#ifndef _foo_
#define _foo_

int foo(int x);

#endif //_foo_

在 gtest 项目中,production_code 库通过引用包含在内 test.cpp

#include "gtest\gtest.h"
#include "gmock\gmock.h"

#include "..\prod\foo.h"

//fake implementation of production code foo
int foo(int x) {
    return x - 1;
}
TEST(TestCaseName, TestName) {
  auto x = foo(5);
  EXPECT_EQ(x, 4);
}

链接器给了我以下错误:

1>prod.lib(foo.obj):错误 LNK2005:“int __cdecl foo(int)” (?foo@@YAHH@Z) 已在 test.obj 1>C:\Example\prod_test.exe 中定义 : 致命错误 LNK1169: 找到一个或多个多重定义的符号

我在这里错过了什么?为什么这不起作用?

如果我将命令“/FORCE:MULTIPLE”添加到链接器,那么我只会收到警告,但我认为这不是正确的做法。

【问题讨论】:

  • foo 的两种不同定义打破了单一定义规则(ODR):en.cppreference.com/w/cpp/language/definition 因此作者提出的技术不受标准支持。 "...每个非内联函数或 odr 使用的变量(见下文)的一个且只有一个定义需要出现在整个程序(包括任何标准和用户定义的库)中。.. 。”
  • 一个函数不能有两个不同的定义。如果你想使用假的,你必须在构建 UT 时从构建系统中排除真正的实现。如果您想正确执行此操作,请使用 GMock 模拟和依赖注入。而 FORCE:MULTIPLE 很可能只是让链接器“随机选择实现”,这不是你想要的。

标签: c++ linker-errors googletest


【解决方案1】:

出现这种情况是因为您在 2 个地方定义了 foo(int)foo.cpptest.cpp,并且您使用这些文件构建了代码。如果您需要使用stub(在这种情况下为假函数foo(int))运行一些测试,您需要创建2个构建目标:

  1. 构建您的真实应用程序
  2. 为单元测试构建一个特殊的应用程序(然后运行它)

当您为单元测试构建应用程序时,您将其与 test.cpp 链接(但不与 foo.cpp 链接)。此外,当您构建您的真实应用程序时,您会将其与 foo.cpp 链接(但不与 test.cpp 链接)。

注意:当这个存根根据测试的想法提供一些额外的功能时,测试真实代码并创建一个存根是有意义的(例如,你检查你的sort()函数,但你可以使用存根为sort() 生成数据,因为数据是在一些复杂的算法完成工作后提供的,计算需要很多时间)或者你不想使用一些资源(例如,你检查你的sort() 函数但是您使用与服务器的网络连接来获取sort() 的真实数据)或者您需要提供一些特定数据来测试您的算法(例如,使用特定数据检查corner case;或者您可能已经找到sort() 不适用于特定数据)。但同样,测试真实代码是有意义的。

【讨论】:

  • 感谢您的回复,如果我理解正确的章节中的书。出于测试目的,链接时间替换故意违反了一项定义规则!在我的解决方案中,我将测试项目与生产项目分开。问题是为什么 Visual Studio 链接器不会将静态库中的定义替换为 test.cpp 中的实现。
  • 也许您将 foo.cpp 添加到单独的测试项目中。在这种情况下,您需要从测试项目中排除 foo.cpp
猜你喜欢
  • 2011-07-02
  • 2021-03-24
  • 1970-01-01
  • 2013-06-30
  • 1970-01-01
  • 2021-08-24
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多