c++ - CMake意外地将最后一个文件名添加到foreach中的可执行文件中

标签 c++ cmake

我正在尝试为我正在处理的图书馆项目创建一个 CMakeLists.txt。这是一个相当简单的结构,我已经能够通过以下设置重新创建一个我遇到的问题的简单示例:

project
├── CMakeLists.txt
├── include
│   ├── doctest.hpp
│   └── foo.hpp
└── src
    ├── foo.cpp
    ├── bar.cpp
    └── test-main.cpp

示例代码在 this gist 中可用(_ 代替了 / 的要点)。 C++ 源代码和 header 实际上不会影响问题,所以我不包括它们的内容,但我的 CMakeLists.txt 实际上是这样的:

cmake_minimum_required(VERSION 3.13)
project(foo VERSION 0.0.1)

set(source_files
        ${CMAKE_SOURCE_DIR}/src/foo.cpp
        ${CMAKE_SOURCE_DIR}/src/bar.cpp
)

set(header_files
        ${CMAKE_SOURCE_DIR}/include/foo.hpp
)

set(DOCTEST_INCLUDE_DIR ${CMAKE_SOURCE_DIR}/include)
add_library(Doctest IMPORTED INTERFACE)
target_include_directories(Doctest INTERFACE ${DOCTEST_INCLUDE_DIR})

add_library(test-main OBJECT ${CMAKE_SOURCE_DIR}/src/test-main.cpp)
target_link_libraries(test-main Doctest)

foreach(src ${source_files})
        # Strip off leading path and extension, leaving only filename without extension to be ${testName}
        string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" "" ${src})
        set(testName ${CMAKE_MATCH_1})
        message(STATUS ${testName})

        add_executable("${testName}" "${src}" $<TARGET_OBJECTS:test-main>)
        target_include_directories(${testName} PRIVATE ${CMAKE_SOURCE_DIR}/include)
        add_test(NAME ${testName} COMMAND ${testName})
endforeach()

问题是尽管 cmake . 运行良好,make 也是如此,而不是输出可执行文件是 src/[filenameWithoutExtensionHere]我希望它是(来自 line 24 in CMakeLists )它最终是 src/bar.cppfoosrc/bar.cppbar,显然将源列表中的最终文件名添加到可执行文件中。

这是由于我在使用 foreach 时或在调用 add_executable 时没有意识到的原因造成的吗? message(STATUS ${testName}) 调用打印出我所期望的,只是没有路径或扩展名的文件名。我还尝试了在 add_executable 中引用/不引用 ${testName} 以及删除 NAMECOMMAND 的各种组合 来自 add_test 但我不认为这是其中的一部分。这是我使用 CMake 的第一个项目,我有点不知道为什么会这样。

导致最终源文件名前置的原因是什么?我该如何避免?

最佳答案

这是由行触发的

string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" "" ${src})

通过指定 ""对于 <output variable> ,你似乎试图忽略它,因为你只关心捕获组......但这似乎有一些非常奇怪和不幸的副作用。您可以使用这个最小的 CMakeLists.txt 获得相同的效果:

cmake_minimum_required(VERSION 3.13)
add_executable(foo ${CMAKE_SOURCE_DIR}/src/foo.cpp)
set("" some_prefix_)

我想这会创建一个名称为空字符串的变量,但是为什么这会影响生成的二进制文件的名称是任何人的猜测。

就放dummymatch_output或者其他什么,问题就消失了:

string(REGEX MATCH "^.*/src/(.*)\\.[^.]*$" match_output ${src})

关于c++ - CMake意外地将最后一个文件名添加到foreach中的可执行文件中,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56755832/

相关文章:

c++ - 为什么字符串 operator= 上的 double free?

c++ - Noise++ Perlin 模块一直返回 0

makefile - 仅当源被先前的命令修改时才重建

c++ - 为什么这个 CMake 脚本找到 "alloca"仍然失败?

c++ - 使用泛型类专门化模板函数

c# - 为什么 C# 没有 C++ 风格的静态库?

c++ - openann C++编译失败

visual-c++ - 如何使用 CMake 在 Visual Studio 2017 RC1 中设置环境变量?

c++ - 在 ubuntu 20.04 上安装 Catch2 #include <catch2/catch.hpp>

android - Gradle 无法使用 OBJECT 库构建 CMake 项目,因为它需要一个输出文件