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

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

相关文章:

  • 前端实习校验函数汇总(未完成)
  • 一个超强的推理增强大模型,开源了,本地部署
  • 驭码CodeRider 2.0 产品体验:在VSCode安装并创建一个雷电小游戏
  • 【Web】腾讯云 COS 静态网站部署与自定义域名 HTTPS 全流程
  • IIS 实现 HTTPS:OpenSSL证书生成与配置完整指南
  • DeepSeek-R1与Claude 4.0 Sonnet:开源与闭源大模型的商业生态博弈
  • Electron-vite【实战】MD 编辑器 -- 大纲区(含自动生成大纲,大纲缩进,折叠大纲,滚动同步高亮大纲,点击大纲滚动等)
  • PyCharm Python IDE
  • 【沉浸式解决问题】Idea运行Junit测试中scanner无法获取控制台的输入内容
  • 创建型设计模式
  • 25-Oracle 23ai DBMS_SEARCH — Ubiquitous Search(无处不在的搜索)
  • 软件架构期末复习
  • WinForm中实现Adobe PDF Reader实现旋转PDF功能
  • 从零打造前沿Web聊天组件:从设计到交互
  • PHP性能提升方案
  • 深入理解 Go 中的字节序(Endianness)检测代码
  • Qt::QueuedConnection详解
  • 前端框架vue3的变化之处
  • 2025 年 MQTT 技术趋势:驱动 AI 与物联网未来发展的关键动力
  • 基于YOLOv12的电力高空作业安全检测:为电力作业“保驾护航”,告别安全隐患!
  • 网站设计公司 -/seo推广收费标准
  • 做国际网站怎么能快速打开/百色seo快速排名
  • 网站开发PHP留言本/aso应用商店优化
  • 网站微信分享链接怎么做的/优化大师在哪里
  • 漳州企业网站建设制作/贵州整站优化seo平台
  • 淘宝网站建设代码/整合营销传播最基础的形式是