Pat*_*cia 5 c++ linux makefile cmake googletest
我承认我有一个独特的情况。我们使用 Make 构建我们的应用程序。但是我的 IDE CLion 使用 CMake。所以我尝试将 GoogleTest 设置为在两者上运行(有点)。我可以两种方式编译我的代码(在命令行中使用 make 并从我的 IDE 中构建)。但是当我选择测试装置并单击运行按钮时,在 CLion 中,没有找到测试,这就是我收到的:
Running main() from gtest_main.cc
[==========] Running 0 tests from 0 test cases.
[==========] 0 tests from 0 test cases ran. (0 ms total)
[ PASSED ] 0 tests.
Process finished with exit code 0
Run Code Online (Sandbox Code Playgroud)
这是我的测试夹具:
#include <gtest/gtest.h>
#include "OPProperties.h"
namespace {
// The fixture for testing class OPPropertiesTestTest.
class OPPropertiesTestTest : public ::testing::Test {
protected:
// You can remove any or all of the following functions if its body
// is empty.
OPPropertiesTestTest() {
// You can do set-up work for each test here.
}
virtual ~OPPropertiesTestTest() {
// You can do clean-up work that doesn't throw exceptions here.
}
// If the constructor and destructor are not enough for setting up
// and cleaning up each test, you can define the following methods:
virtual void SetUp() {
// Code here will be called immediately after the constructor (right
// before each test).
}
virtual void TearDown() {
// Code here will be called immediately after each test (right
// before the destructor).
}
// Objects declared here can be used by all tests in the test case for OPPropertiesTestTest.
};
TEST_F(OPPropertiesTestTest, ThisTestWillPass) {
EXPECT_EQ(0, 0);
}
TEST_F(OPPropertiesTestTest, ThisTestWillFail) {
EXPECT_EQ(0, 5);
}
} // namespace
int main(int argc, char **argv) {
::testing::InitGoogleTest(&argc, argv);
return RUN_ALL_TESTS();
}
Run Code Online (Sandbox Code Playgroud)
这是我的 CMakeLists.txt 文件:
cmake_minimum_required(VERSION 2.8)
project(oneprint)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -pthread -lgtest")
add_definitions(-DBOOST_LOG_DYN_LINK)
set(SOURCE_FILES
src/controllers/OPProperties.cpp
src/controllers/OPProperties.h
src/main.cpp)
include_directories(src/controllers)
set(BOOST_ROOT "/usr/local/lib")
set(Boost_USE_STATIC_LIBS OFF)
set(Boost_USE_MULTITHREADED ON)
unset(Boost_INCLUDE_DIR CACHE)
unset(Boost_LIBRARY_DIRS CACHE)
#set(Boost_LIBRARY_DIR /usr/local/arm/lib)
set(OpenCV_LIBRARY_DIR /usr/include/opencv2)
set(Innovatrics_LIBRARY_DIR /usr/local/arm/lib)
find_package(OpenCV REQUIRED)
find_package(Boost 1.57.0 COMPONENTS filesystem thread log chrono system atomic program_options REQUIRED)
include_directories(${Boost_INCLUDE_DIRS})
enable_testing()
find_package(GTest REQUIRED)
include_directories(${GTEST_INCLUDE_DIRS})
add_executable(OnePrint ${SOURCE_FILES})
target_link_libraries(OnePrint ${OpenCV_LIBS})
target_link_libraries(OnePrint ${Boost_LIBRARIES})
target_link_libraries(OnePrint ${Innovatrics_LIBRARY_DIR})
target_link_libraries(OnePrint ${GTEST_BOTH_LIBRARIES})
target_link_libraries(OnePrint pthread)
Run Code Online (Sandbox Code Playgroud)
我在 src 下添加了一个名为 tests 的文件夹,这是我的测试装置 OPPropertiesTestTest 所在的位置。我还在顶层添加了一个测试文件夹。此文件夹中包含一个 Makefile 和一个 Srcs.mak 文件。
这是生成文件:
TARGET = oneprint
BASE = ../
-include $(BASE)Defs.x86.mak
-include $(BASE)OpenCV.mak
-include $(BASE)Boost.mak
-include $(BASE)Innovatrics.mak
-include $(BASE)GTest.mak
-include $(BASE)Incl.mak
-include Srcs.mak
-include $(BASE)Common.mak
-include $(BASE)App.mak
Run Code Online (Sandbox Code Playgroud)
这是 Srcs.mak 文件:
VPATH = \
../src/controllers:\
../src:\
../src/tests
CPP_SRCS = \
OPProperties.cpp \
# test files
CPP_SRCS += \
OPPropertiesTest.cpp
Run Code Online (Sandbox Code Playgroud)
我通常不希望看到 makefile 被混合到 CMake 管理的项目中;听起来你的设置很复杂?
除此之外,我认为这里的根本原因可能是双重的。我认为/src/tests/实际上并没有构建测试可执行文件。我在你的 CMakelists.txt 中看不到任何会导致它被构建的东西,所以除非你做了一些你没有向我们展示的额外内容,否则不会编译测试文件。
可能使它们看起来像的是,您正在将 gtest 提供的帮助程序库链接到您的OnePrint目标中。这个助手库的不寻常之处在于它实现了一个main()功能来保存用户必须指定自己的功能。
你正在做这行
target_link_libraries(OnePrint ${GTEST_BOTH_LIBRARIES})
Run Code Online (Sandbox Code Playgroud)
来自文档的 GTEST_BOTH_LIBRARIES是一个包含 libgtest 和 libgtest-main 的变量。您只需要 libgtest,因为您已经编写了自己的main(). 所以你应该GTEST_LIBRARIES改用。
注意:CMake 代码还有一些其他问题:
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -pthread -lgtest")
Run Code Online (Sandbox Code Playgroud)
这不是跨平台的(这对您来说可能无关紧要),例如 MSVC 不会识别任何这些标志。如果确实重要,您需要将其包装在if(UNIX)orif(NOT MSVC)块中。
此外,这会全局设置编译器标志,即为项目中定义的每个目标设置。如果您需要更细粒度的控制,请查看target_compile_options哪个可以让您为每个目标指定这些。
另一个问题是-lgtest零件。这实际上是调用链接器来链接 gtest 库。但是,CMake 会为您处理这些问题 - 您在调用target_link_libraries(OnePrint ${GTEST_LIBRARIES}).
add_definitions(-DBOOST_LOG_DYN_LINK)
Run Code Online (Sandbox Code Playgroud)
虽然这很好,但关于在全球范围内应用的相同评论也适用。每个目标的等价物是target_compile_definitions。
set(BOOST_ROOT "/usr/local/lib")
Run Code Online (Sandbox Code Playgroud)
这可能是 Boost 在您机器上的位置,但对于其他开发人员来说可能并非如此。该变量实际上是由单个用户在调用 CMake 时指定的(如果需要……CMake 会自动搜索许多路径)——您不应该将这样的路径硬编码到 CMake 文件中。
target_link_libraries(OnePrint pthread)
Run Code Online (Sandbox Code Playgroud)
同样,pthread如果您使用 MSVC ,则不适合链接;这可能应该包裹在一个if块中。
最后一点是您可以在target_link_libraries命令中指定多个依赖项,因此您可以将它们更改为:
if(NOT MSVC)
set(PThreadLib pthread)
endif()
target_link_libraries(OnePrint
${OpenCV_LIBS}
${Boost_LIBRARIES}
${Innovatrics_LIBRARY_DIR}
${GTEST_LIBRARIES}
${PThreadLib})
Run Code Online (Sandbox Code Playgroud)