与 googletest 链接时未定义的引用

use*_*014 6 makefile cmake static-libraries googletest

当我尝试构建 googletest(和 googlemock)1.8.0 时,当我尝试与 libgtest.a 链接时,我得到了对 MakeAndRegisterTestInfo 的未定义引用。它适用于具有相同 cmake/make 设置的 1.7.0 版。我想我可以使用 1.7.0,但是我需要单独下载和构建 gmock。

CMakeFiles/unittest.dir/test/test_led.cpp.o: In function `__static_initialization_and_destruction_0(int, int)':
    test_led.cpp:(.text+0x23d): undefined reference to `testing::internal::MakeAndRegisterTestInfo(char const*, char const*, char const*, char const*, void const*, void (*)(), void (*)(), testing::internal::TestFactoryBase*)'
    collect2: error: ld returned 1 exit status
    make[2]: *** [bin/unittest] Error 1
    make[1]: *** [CMakeFiles/unittest.dir/all] Error 2
    make[1]: *** Waiting for unfinished jobs....
Run Code Online (Sandbox Code Playgroud)

libgtest.a 中的其他符号工作得很好(例如 ::testing::InitGoogleTest),但是一旦我尝试使用宏 TEST_F 添加测试,我就会收到此错误。

这是我的测试用例设置:

#include "gtest/gtest.h"

namespace {
// The fixture for testing (used by TEST_F).
class Foo : public ::testing::Test {
protected:
    Foo();
    virtual ~Foo() {};    
    virtual void SetUp();
    virtual void TearDown();
};

Foo::Foo() {
};

void Foo::SetUp()
{
};

void Foo::TearDown()
{
};

TEST_F(Foo, Init) {
};
} // namespace

int main(int argc, char **argv)
{
    ::testing::InitGoogleTest(&argc, argv);
    int ret = RUN_ALL_TESTS();
    return ret;
}
Run Code Online (Sandbox Code Playgroud)

Cmake google 测试片段:它从本地文件夹下载。它构建,我得到一个 libgtest.a 文件。

### Unit test ###
# Google test
ExternalProject_Add(EXT_googletest
                    PREFIX ${CMAKE_CURRENT_BINARY_DIR}
                    DOWNLOAD_COMMAND ""
                    SOURCE_DIR ${CMAKE_SOURCE_DIR}/../external/googletest
                    BUILD_COMMAND make all
                    # Disable install step
                    INSTALL_COMMAND "")
# Create a libgtest target to be used as a dependency by test programs
set(LIBGTEST_STATIC ${CMAKE_CURRENT_BINARY_DIR}/.../googlemock/gtest/libgtest.a)

add_executable(unittest
              "test/test_foo.cpp")
target_link_libraries(unittest main_app_lib
                               ${LIBGTEST_STATIC}
                               ${CMAKE_THREAD_LIBS_INIT})
Run Code Online (Sandbox Code Playgroud)

NM 输出 libgtest.a 1.8.0

user@system$ nm .../libgtest.a |grep MakeAndRegisterTestInfo
000000000000743a T _ZN7testing8internal23MakeAndRegisterTestInfoEPKcS2_S2_S2_NS0_12CodeLocationEPKvPFvvES7_PNS0_15TestFactoryBaseE
Run Code Online (Sandbox Code Playgroud)

NM 输出 libgtest.a 1.7.0(工作文件)

user@system$ nm .../libgtest.a |grep MakeAndRegisterTestInfo
0000000000005968 T _ZN7testing8internal23MakeAndRegisterTestInfoEPKcS2_S2_S2_PKvPFvvES6_PNS0_15TestFactoryBaseE
Run Code Online (Sandbox Code Playgroud)

解决了:

在构建机器上安装了旧版本的谷歌测试,cmake 使用了这些头文件。

nmg*_*eek 5

@user1178014 回答了他自己的问题,但由于没有发布答案,我正在写:

如果您直接下载了 gtest 源代码并使用make install了下载的 gtest 存储库中的/usr/local/include/gtest. 如果您以后用于apt-get安装libgtest-devdebian 软件包,它会在/usr/include/gtest. 如果从 debian 包安装的版本较新,/usr/include即使您正确链接了新libgtest.a存档,您的 Makefile 也可以从中获取旧的头文件并给出链接错误。

解决方案是寻找/usr/local/include/gtest/usr/include/gtest查看它们是否都存在。如果他们这样做,则删除旧目录。如果/usr/include/gtest是旧目录,您可能希望通过卸载libgtest-dev软件包来删除它。