繁体   English   中英

捆绑 static C++ 库时如何防止 CMake 双重编译源?

[英]How to prevent CMake from double compiling sources when bundling static C++ libraries?

我正在尝试使用 CMake 构建一个 static 库libbar libbar应该包含libfoo ,即子目录目标libfoo中的所有 object 文件也应该出现在libbar中。 最简单的目录树如下:

bar
├── bar.cpp
├── CMakeLists.txt
└── foo
    ├── CMakeLists.txt
    └── foo.cpp

这是foo/CMakeLists.txt

cmake_minimum_required(VERSION 3.14)
project(foo)

set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

add_library(foo)
target_sources(foo PUBLIC foo.cpp)

这是顶级CMakeLists.txt

cmake_minimum_required(VERSION 3.14)

set(CMAKE_CXX_STANDARD 11)
set(CMAKE_CXX_STANDARD_REQUIRED ON)

project(bar)

add_library(bar)
add_subdirectory(foo)
target_sources(bar PUBLIC bar.cpp)
target_link_libraries(bar PRIVATE foo)

bar/我执行以下操作:

cmake . -Bbuild
cd build
cmake --build .

我得到

Scanning dependencies of target foo
[ 20%] Building CXX object foo/CMakeFiles/foo.dir/foo.cpp.o
[ 40%] Linking CXX static library libfoo.a
[ 40%] Built target foo
Scanning dependencies of target bar
[ 60%] Building CXX object CMakeFiles/bar.dir/bar.cpp.o
[ 80%] Building CXX object CMakeFiles/bar.dir/foo/foo.cpp.o
[100%] Linking CXX static library libbar.a
[100%] Built target bar

如您所见,文件foo.cpp被编译了两次,我正试图摆脱这种行为。 顺便说一句,这个方法给了我一个正确的结果:

$ ar t libbar.a 
bar.cpp.o
foo.cpp.o

如果我在foo/CMakeLists.txt中将 PUBLIC 更改为 PRIVATE ,则构建日志如下:

Scanning dependencies of target foo
[ 25%] Building CXX object foo/CMakeFiles/foo.dir/foo.cpp.o
[ 50%] Linking CXX static library libfoo.a
[ 50%] Built target foo
Scanning dependencies of target bar
[ 75%] Building CXX object CMakeFiles/bar.dir/bar.cpp.o
[100%] Linking CXX static library libbar.a
[100%] Built target bar

但 foo.cpp.o 没有进入libbar

$ ar t libbar.a 
bar.cpp.o

在没有双重编译的情况下构建包含libfoolibbar的正确方法是什么?

target_sources(foo PUBLIC foo.cpp)

此行强制链接到foo的目标在其源中包含foo.cpp

在没有双重编译的情况下构建包含 libfoo 的 libbar 的正确方法是什么?

你已经明确要求这个,所以只是......不要:

target_sources(foo PRIVATE foo.cpp)
  • PUBLIC的意思是“适用于自己和被链接者”。
  • PRIVATE的意思是“适用于自己”
  • INTERFACE的意思是“仅适用于被链接者”

如果您真的希望foo.o在两个档案中(这是可疑的),那么您可以使用libbarlibfoo链接到的OBJECT库。

add_library(foo_objs OBJECT foo.cpp)

# later ...
target_link_libraries(foo PRIVATE foo_objs)
# ...
target_link_libraries(bar PRIVATE foo_objs)

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM