CMakeLists中 get_filename_component详解

get_filename_component是 CMake 中用于路径处理的核心指令,主要作用是提取当前源码目录的上级目录路径并赋值给指定变量。

当我们使用的命令是

get_filename_component(PACKAGE_SOURCE_ROOT ${CMAKE_CURRENT_SOURCE_DIR} PATH)

假设你的项目目录结构如下:

plaintext 复制代码
  
/my_project/                # 根目录(我们要获取的目标目录)
├── src/                    # 子目录
│   └── module/             # 子子目录(当前 CMakeLists.txt 所在目录)
│       └── CMakeLists.txt  # 执行这行代码的文件
└── include/
 

此时:

  • CMAKE_CURRENT_SOURCE_DIR 的值是 /my_project/src/module ;
  • 执行 get_filename_component(PACKAGE_SOURCE_ROOT ${CMAKE_CURRENT_SOURCE_DIR} PATH) 后;
  • PACKAGE_SOURCE_ROOT 的值是 /my_project/src (即当前目录的上级目录)。

如果想获取根目录( /my_project ),可以嵌套执行:

cmake 复制代码
  
# 第一步:拿到 /my_project/src
get_filename_component(PACKAGE_SOURCE_ROOT ${CMAKE_CURRENT_SOURCE_DIR} PATH)
# 第二步:拿到 /my_project(再往上一级)
get_filename_component(PACKAGE_SOURCE_ROOT ${PACKAGE_SOURCE_ROOT} PATH)
message(STATUS "PACKAGE_SOURCE_ROOT = ${PACKAGE_SOURCE_ROOT}")
# 输出:-- PACKAGE_SOURCE_ROOT = /my_project
 

现代替代写法(CMake 3.20+)

CMake 3.20 及以上版本提供了更直观的 cmake_path 指令,功能等价且可读性更好,推荐优先使用:

cmake 复制代码
  
# 等价于原代码:提取上级目录
cmake_path(GET CMAKE_CURRENT_SOURCE_DIR PARENT_PATH PACKAGE_SOURCE_ROOT)
# 如果要提取多级上级目录(比如两级),可以这样写:
cmake_path(GET CMAKE_CURRENT_SOURCE_DIR PARENT_PATH temp_dir)
cmake_path(GET temp_dir PARENT_PATH PACKAGE_SOURCE_ROOT)

常见使用场景

  1. 统一引用上级目录的文件:比如当前子目录的 CMakeLists.txt 需要引用上级目录的 include 文件夹、配置文件等,通过这个变量可以避免写死相对路径(如 .../.../include ),提升代码可移植性。
cmake 复制代码
  
# 获取上级目录
get_filename_component(PACKAGE_SOURCE_ROOT ${CMAKE_CURRENT_SOURCE_DIR} PATH)
# 引用上级目录的 include 文件夹
include_directories(${PACKAGE_SOURCE_ROOT}/include)
  1. 定位项目根目录:通过多次提取上级目录,最终定位到项目根目录,方便统一管理资源、库文件等。
  2. 避免硬编码路径:相比直接写 .../ 这种相对路径,用 get_filename_component 更清晰,且能自动处理不同系统的路径分隔符(Windows 的 \ 和 Linux/macOS 的 / )。

总结

  1. 核心作用: get_filename_component(变量 路径 PATH) 是提取指定路径的上级目录,并赋值给变量,等价于 dirname 命令。
  2. CMAKE_CURRENT_SOURCE_DIR 是当前 CMakeLists.txt 所在的源码目录,结合 PATH 参数就能拿到它的父目录。
  3. 现代写法(CMake 3.20+)推荐用 cmake_path(GET ... PARENT_PATH ...) ,语义更清晰,路径处理更健壮。
相关推荐
魑魅魍魉都是鬼1 天前
Java 适配器模式(Adapter Pattern)
java·开发语言·适配器模式
笨笨马甲1 天前
Qt MQTT
开发语言·qt
Fairy要carry1 天前
面试-Agent上下文过载、步骤混乱的问题
开发语言·python
程序员Ctrl喵1 天前
异步编程:Event Loop 与 Isolate 的深层博弈
开发语言·flutter
liuyao_xianhui1 天前
优选算法_两数之和_位运算_C++
java·开发语言·数据结构·c++·算法·链表·动态规划
IT猿手1 天前
MATLAB模拟四旋翼无人机飞行,机翼可独立旋转
开发语言·matlab·无人机
代龙涛1 天前
WordPress 主题开发指南:模板文件、函数与页面选型规则
开发语言·后端·php·wordpress
代码探秘者1 天前
【大模型应用】6.RAG 场景下的向量+关键词混合检索
java·开发语言·人工智能·python·spring
RFCEO1 天前
JavaScript基础课程十四、原型与原型链(JS 核心底层)
开发语言·原型模式·prototype原型详解·javascript基础课·构造函数原型方法定义与使用·js原型链继承机制入门·t原型链顶层null原理