c++ - 使用 cmake 时如何在 C++ 中使用库(*.a 文件)
问题描述
所以最近我又开始编写 C++ 了。我想学习如何正确使用 Cmake,并从现在开始为我的爱好项目编写适当的测试。
我基本上只是想将 GTest 集成到我的项目中。我设置了一个 TestProject 目录并执行了以下操作:
- 克隆了GTest 存储库
- 使用 cmake 在 googletest 存储库中构建代码
googletest/build/lib
将目录复制到TestProject/
googletest/include
将目录复制到TestProject/
现在,当我尝试使用 cmake 时,链接时出现以下错误:
/usr/bin/ld: ./libgtest.a(gtest-all.cc.o): in function `testing::internal::ThreadLocal<testing::TestPartResultReporterInterface*>::~ThreadLocal()':
gtest-all.cc:(.text._ZN7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEED2Ev[_ZN7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEED5Ev]+0x24): undefined reference to `pthread_getspecific'
/usr/bin/ld: gtest-all.cc:(.text._ZN7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEED2Ev[_ZN7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEED5Ev]+0x39): undefined reference to `pthread_key_delete'
/usr/bin/ld: ./libgtest.a(gtest-all.cc.o): in function `testing::internal::ThreadLocal<std::vector<testing::internal::TraceInfo, std::allocator<testing::internal::TraceInfo> > >::~ThreadLocal()':
gtest-all.cc:(.text._ZN7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEED2Ev[_ZN7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEED5Ev]+0x24): undefined reference to `pthread_getspecific'
/usr/bin/ld: gtest-all.cc:(.text._ZN7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEED2Ev[_ZN7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEED5Ev]+0x39): undefined reference to `pthread_key_delete'
/usr/bin/ld: ./libgtest.a(gtest-all.cc.o): in function `testing::internal::ThreadLocal<std::vector<testing::internal::TraceInfo, std::allocator<testing::internal::TraceInfo> > >::GetOrCreateValue() const':
gtest-all.cc:(.text._ZNK7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEE16GetOrCreateValueEv[_ZNK7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEE16GetOrCreateValueEv]+0x25): undefined reference to `pthread_getspecific'
/usr/bin/ld: gtest-all.cc:(.text._ZNK7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEE16GetOrCreateValueEv[_ZNK7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEE16GetOrCreateValueEv]+0x88): undefined reference to `pthread_setspecific'
/usr/bin/ld: ./libgtest.a(gtest-all.cc.o): in function `testing::internal::ThreadLocal<testing::TestPartResultReporterInterface*>::CreateKey()':
gtest-all.cc:(.text._ZN7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEE9CreateKeyEv[_ZN7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEE9CreateKeyEv]+0x27): undefined reference to `pthread_key_create'
/usr/bin/ld: ./libgtest.a(gtest-all.cc.o): in function `testing::internal::ThreadLocal<std::vector<testing::internal::TraceInfo, std::allocator<testing::internal::TraceInfo> > >::CreateKey()':
gtest-all.cc:(.text._ZN7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEE9CreateKeyEv[_ZN7testing8internal11ThreadLocalISt6vectorINS0_9TraceInfoESaIS3_EEE9CreateKeyEv]+0x27): undefined reference to `pthread_key_create'
/usr/bin/ld: ./libgtest.a(gtest-all.cc.o): in function `testing::internal::ThreadLocal<testing::TestPartResultReporterInterface*>::GetOrCreateValue() const':
gtest-all.cc:(.text._ZNK7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEE16GetOrCreateValueEv[_ZNK7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEE16GetOrCreateValueEv]+0x25): undefined reference to `pthread_getspecific'
/usr/bin/ld: gtest-all.cc:(.text._ZNK7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEE16GetOrCreateValueEv[_ZNK7testing8internal11ThreadLocalIPNS_31TestPartResultReporterInterfaceEE16GetOrCreateValueEv]+0x88): undefined reference to `pthread_setspecific'
我的项目目录中的文件是:
TestProject/CMakeLists.txt
cmake_minimum_required(VERSION 3.16)
project(TestGoogleTest)
set(ROOT_DIR "${PROJECT_SOURCE_DIR}")
set(SRC "${ROOT_DIR}/src")
set(LIB "${ROOT_DIR}/lib")
set(INCLUDE "${ROOT_DIR}/include")
link_directories("${LIB}")
include_directories("${INCLUDE}")
message("INCLUDE PATH: ${INCLUDE}")
set(SOURCES "${SRC}/main.cpp" "${INCLUDE}/gtest/gtest.h")
add_executable(main "${SOURCES}")
target_link_libraries(main libgtest.a)
TestProject/src/main.cpp
:
#include "gtest/gtest.h"
#include <iostream>
int main(int argc, char **argv)
{
testing::InitGoogleTest(&argc, argv);
std::cout << "Hello world!" << std::endl;
return RUN_ALL_TESTS();
}
如果我使用系统 gtest 使用命令手动编译它,它工作正常
g++ src/main.cpp -o main -lgtest
我很高兴了解导致问题的原因:)
解决方案
简短说明
在你的CMakeLists.txt
- 您尚未设置
c++11
GoogleTest 需要的标志 - 您没有链接
pthread
库,因此 GoogleTest 找不到与此相关的定义
长解释
尽管有一些更好的方法可以实现相同的目标,但我将坚持您为您的应用程序所遵循的构建策略。
我假设以下目录结构
├── CMakeLists.txt
├── include
│ └── gtest
│ ├── gtest-death-test.h
│ ├── gtest.h
│ ├── gtest-matchers.h
│ ├── gtest-message.h
│ ├── gtest-param-test.h
│ ├── gtest_pred_impl.h
│ ├── gtest-printers.h
│ ├── gtest_prod.h
│ ├── gtest-spi.h
│ ├── gtest-test-part.h
│ ├── gtest-typed-test.h
│ └── internal
│ ├── custom
│ │ ├── gtest.h
│ │ ├── gtest-port.h
│ │ ├── gtest-printers.h
│ │ └── README.md
│ ├── gtest-death-test-internal.h
│ ├── gtest-filepath.h
│ ├── gtest-internal.h
│ ├── gtest-param-util.h
│ ├── gtest-port-arch.h
│ ├── gtest-port.h
│ ├── gtest-string.h
│ └── gtest-type-util.h
├── libs
│ ├── libgmock.a
│ ├── libgmock_main.a
│ ├── libgtest.a
│ └── libgtest_main.a
├── main
└── src
└── main.cpp
然后我g++
直接从 shell 调用,输入以下行(试图模仿你的调用策略)
g++ src/main.cpp --std=c++11 -o main -Iinclude -Llib -lgtest -pthread
旗帜的含义
- GoogleTest 需要
c++11
兼容的编译器,因此我们需要--std=c++11
标志 -Iinclude
需要在include
不指定include
目录名称的情况下使用标题-Llib
gtest
指定库所在的路径pthread
需要使用 GoogleTest,尽管可以构建 GoogleTest 使其可以以单线程方式使用
它可以正常工作而不会引起任何问题
然后我模仿了你CMakeLists.txt
并添加了一些下面突出显示的东西(没有任何问题)
cmake_minimum_required(VERSION 3.16)
project(TestGoogleTest)
# added the line below to find libraries for threading
find_package(Threads)
set(CMAKE_CXX_STANDARD 11)
set(ROOT_DIR "${PROJECT_SOURCE_DIR}")
set(SRC "${ROOT_DIR}/src")
set(LIB "${ROOT_DIR}/lib")
# added the line below to enable c++11 features
set(INCLUDE "${ROOT_DIR}/include")
link_directories("${LIB}")
include_directories("${INCLUDE}")
message("INCLUDE PATH: ${INCLUDE}")
set(SOURCES "${SRC}/main.cpp" "${INCLUDE}/gtest/gtest.h")
add_executable(main "${SOURCES}")
# modified the line below to express link dependency on gtest and thread libraries
# lib prefix can be omitted here
target_link_libraries(main gtest ${CMAKE_THREAD_LIBS_INIT})
推荐阅读
- reactjs - Redux 存储在集成测试期间未更新
- javascript - 来自子组件的 Vue ref 调用方法返回未定义的方法(v-for)
- r - R - IMDb 数据集文件 - 如何合并每部电影的线条
- python - 如何在 python 中使用 selenium 循环遍历这些元素并获得 XPATH 值作为回报?
- ansible - 在 ansible play 中获取当前失败/跳过/正常任务的数量
- sql - 从 AVG 中删除 Null 结果
- d3.js - 表 d3.js 中 div 元素中的交互式注释
- python - 从数据框单元格中的字符串值中删除“空”值
- mysql - 如何从多选中选择行的索引
- geoserver - 通过脚本设置地理服务器配置