当前位置: 首页 > ai >正文

CMake指令: add_sub_directory以及工作流程

目录

1.简介

2.工作流程

3.示例场景

4.最佳实践

5.注意事项

6.总结

相关链接


1.简介

   add_subdirectory 是 CMake 中用于添加子目录参与构建的命令,允许将项目拆分为多个模块或子项目,实现代码的模块化管理。

        基本语法:

add_subdirectory(source_dir [binary_dir] [EXCLUDE_FROM_ALL])
  • source_dir:子目录的源代码路径(相对于当前 CMakeLists.txt 的路径)。
  • binary_dir(可选):指定子目录的编译输出路径(默认与 source_dir 同级的 build 目录)。
  • EXCLUDE_FROM_ALL(可选):子目录不会被默认构建,需显式调用 add_subdirectory 或指定目标依赖。

        核心作用:

  1. 模块化构建:将项目拆分为多个子目录(如 srcteststhird_party),每个子目录包含独立的 CMakeLists.txt
  2. 依赖管理:子目录可定义库或可执行文件,供父目录或其他子目录链接。
  3. 递归构建:子目录中的 add_subdirectory 会被递归处理,实现多层级项目结构。

2.工作流程

1.目录解析

  • CMake 解析 add_subdirectory() 中的 source_dir 参数(如 src),确定子目录的路径。
  • 若指定 binary_dir(如 build/src),则将子目录的构建输出定向到该路径。
# 父目录 CMakeLists.txt
add_subdirectory(src)  # 无 binary_dir,输出到 build/src

2.变量传递:

  • 父→子传递:父目录中的变量(如 CMAKE_CXX_FLAGSPROJECT_NAME)自动传递给子目录。
  • 子→父传递:子目录可通过 set(... PARENT_SCOPE) 将变量回传给父目录。
# 父目录定义
set(COMMON_FLAGS "-Wall")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${COMMON_FLAGS}")# 子目录自动继承 COMMON_FLAGS 和 CMAKE_CXX_FLAGS# 子目录
set(MY_VERSION "1.0.0" PARENT_SCOPE)  # 传递到父目录# 父目录
message("Version from subdir: ${MY_VERSION}")  # 输出 1.0.0

3.子目录处理

  • CMake 递归执行子目录中的 CMakeLists.txt 文件,生成目标(如 add_libraryadd_executable)。
  • 子目录中的 add_subdirectory() 会被递归处理,形成构建树。
# src/CMakeLists.txt
add_library(my_lib STATIC src/file.cpp)
target_include_directories(my_lib PUBLIC include)

4.依赖关系建立

  • 子目录中定义的目标(如 lib)可被父目录或其他子目录链接(如 target_link_libraries)。
  • CMake 自动处理目标间的依赖关系,确保正确的构建顺序。
# 父目录 CMakeLists.txt
add_subdirectory(src)  # 先处理子目录,生成 my_libadd_executable(main main.cpp)
target_link_libraries(main PRIVATE my_lib)  # 链接子目录的库

3.示例场景

假设你的项目结构如下:

MyProject/
├── CMakeLists.txt
├── src/
│   ├── CMakeLists.txt
│   ├── main.cpp
│   ├── utils.cpp
│   └── app.cpp
├── lib/
│   ├── CMakeLists.txt
│   ├── lib1.cpp
│   └── lib2.cpp
└── include/├── utils.h└── app.h

主目录的 CMakeLists.txt

cmake_minimum_required(VERSION 3.10)
project(MyProject)# 设置C++标准
set(CMAKE_CXX_STANDARD 17)
set(CMAKE_CXX_STANDARD_REQUIRED True)# 添加包含目录
include_directories(${PROJECT_SOURCE_DIR}/include)# 添加子目录
add_subdirectory(lib)
add_subdirectory(src)# 定义最终的可执行文件,并链接子目录生成的库
add_executable(MyApp ${SRC_FILES})
target_link_libraries(MyApp PRIVATE mylib)# 输出配置信息
message(STATUS "Source files: ${SRC_FILES}")
message(STATUS "Library files: ${LIB_SOURCES}")

lib 子目录的 CMakeLists.txt

# lib/CMakeLists.txt# 定义库的源文件列表
set(LIB_SOURCESlib1.cpplib2.cpp
)# 创建静态库或动态库
add_library(mylib STATIC ${LIB_SOURCES})
# 或者创建动态库
# add_library(mylib SHARED ${LIB_SOURCES})# 指定库的包含目录
target_include_directories(mylib PUBLIC ${PROJECT_SOURCE_DIR}/include)# 将库源文件传递到父作用域
set(LIB_SOURCES ${LIB_SOURCES} PARENT_SCOPE)

src 子目录的 CMakeLists.txt

# src/CMakeLists.txt# 定义源文件列表,包含当前目录的源文件
set(SRC_FILESmain.cpputils.cppapp.cpp
)# 将源文件传递到父作用域
set(SRC_FILES ${SRC_FILES} PARENT_SCOPE)

4.最佳实践

1.项目结构建议

project/
├─ CMakeLists.txt          # 根目录:设置全局变量、添加子目录
├─ include/                # 公共头文件
├─ src/
│  ├─ CMakeLists.txt       # 定义库或可执行文件
│  └─ ...                  # 源代码
├─ tests/
│  ├─ CMakeLists.txt       # 测试相关目标
│  └─ ...                  # 测试代码
└─ third_party/            # 第三方依赖(可选)

2.模块化管理

在子目录中封装功能模块(如 add_library),通过 target_* 命令暴露接口,避免全局变量污染。

# 子目录 src/CMakeLists.txt
add_library(utils STATIC utils.cpp)
target_include_directories(utils PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/include)

3.条件编译与选项

使用 option 或 if 控制子目录是否参与构建,提升灵活性:

# 根据选项决定是否添加测试子目录
option(BUILD_TESTS "Build tests" ON)
if(BUILD_TESTS)add_subdirectory(tests)
endif()

4.排除默认构建(EXCLUDE_FROM_ALL)

# 子目录不会被默认构建,需显式依赖(如通过 add_dependencies)
add_subdirectory(third_party EXCLUDE_FROM_ALL)

5.注意事项

1.变量作用域问题

  • 子目录无法直接修改父目录的变量,需通过 PARENT_SCOPE 回传。
  • 避免在子目录中使用全局变量(如 include_directories),改用 target_* 命令。
set(VERSION "1.0" PARENT_SCOPE)  # 子目录回传变量到父目录

2.多级子目录

如果项目中有多级子目录,例如 src/module1 和 src/module2,可以在 src/CMakeLists.txt 中进一步使用 add_subdirectory(module1) 和 add_subdirectory(module2) 来递归处理这些子目录。

# src/CMakeLists.txtadd_subdirectory(module1)
add_subdirectory(module2)# 定义 src 目录下的源文件
set(SRC_FILESmain.cpputils.cppapp.cpp
)# 将源文件传递到父作用域
set(SRC_FILES ${SRC_FILES} PARENT_SCOPE)

3.构建顺序

  • add_subdirectory 的调用顺序决定子目录的处理顺序,但目标的构建顺序需通过 target_link_libraries 或 add_dependencies 显式指定。

4.使用相对路径和全局变量

        在子目录的 CMakeLists.txt 中,路径通常是相对于子目录本身的。例如,lib/CMakeLists.txt 中的 lib1.cpp 实际上指的是 lib/lib1.cpp。

        如果需要在多个子目录中共享变量或路径,可以在主目录中定义全局变量或使用 CMake 的全局范围选项(如 CACHE 变量)来传递信息。

5.错误处理

        如果 add_subdirectory() 指定的子目录不存在或没有 CMakeLists.txt 文件,CMake 会报错并中止配置过程。因此,确保所有子目录中都存在有效的 CMakeLists.txt 文件。

6.总结

     add_subdirectory()的优点:

  • 结构清晰:项目目录层次分明,便于导航和理解。
  • 模块化:每个模块或组件可以独立开发和测试。
  • 灵活性:每个子目录可以有不同的编译选项和依赖关系。
  • 可扩展性:轻松添加新的模块或组件,无需修改主 CMakeLists.txt

   add_subdirectory() 的核心价值在于实现项目的模块化构建,通过合理拆分代码和分层管理 CMakeLists.txt,可显著提升大型项目的可维护性。充分利用 CMake 提供的命令和功能,如 target_include_directories()target_link_libraries() 等,来管理依赖关系和编译选项。

相关链接

  • CMake 官网 CMake - Upgrade Your Software Build System
  • CMake 官方文档:CMake Tutorial — CMake 4.0.2 Documentation
  • CMake 源码:https://github.com/Kitware/CMake
  • CMake 源码:CMake · GitLab
  • 中文版基础介绍: CMake 入门实战 | HaHack
  • wiki: Home · Wiki · CMake / Community · GitLab
http://www.xdnf.cn/news/13506.html

相关文章:

  • 速盾:高防CDN可以加速数据库吗?
  • ​​5G通信设备线路板打样:猎板PCB如何攻克高速数据传输技术瓶颈​​
  • bat 批处理查看文件年龄
  • C51 KEIL使用使用问题处理
  • Java异步编程深度解析:从基础到复杂场景的难题拆解
  • K8S中应用无法获取用户真实ip问题排查
  • 数据链抗干扰
  • DNS小结
  • 避免在 iOS 和 Android 的 WebView 中长按出现复制框等默认行为
  • 手机解压 7z 文件全攻略
  • 【全志V821_FoxPi】2-2 切换为spi nand方案启动
  • HTML5 浮动
  • 统计可分解整数的数量
  • leetcode1584. 连接所有点的最小费用-medium
  • 2025低空经济区的安全与应急控制专题研讨会(SECOLZ 2025)
  • DDoS攻防实战:从应急脚本到AI云防护系统
  • 2025年智慧城市与管理工程国际会议(ICSCME 2025)
  • 第二章——线性表之循环链表、静态链表
  • 机械ERP需要解决的几个问题?关于非标机械行业物料编码,如何提升建立效率的说明!
  • 【深度学习】深度学习中的张量:从多维数组到智能计算单元
  • GO语言使用gorm的dbresolver插件实现数据库读写分离
  • iOS开发申请组播/广播权限​
  • 【C/C++】long long 类型传参推荐方式
  • asio之静态互斥量
  • 【PmHub面试篇】集成 Sentinel+OpenFeign实现网关流量控制与服务降级相关面试题解答
  • 远程io模块在汽车流水线的应用
  • 深度学习工具四剑客:Anaconda、Jupyter、PyTorch与CUDA详解
  • 达梦数据库dsc集群+异步主备
  • DeviceNet转Modbus RTU网关在玻璃制造中的关键应用
  • 如何制定兼容多个项目的整体时间计划?