CMake中有两个变量,可以获取到当前CMakeLists.txt
的当前目录名(绝对路径)和当前文件的绝对路径,分别是CMAKE_CURRENT_SOURCE_DIR
和CMAKE_CURRENT_LIST_FILE
。
但是,有时候需要的往往是相对路径名。比如我们编写项目的一些样例,以当前文件夹的名字作为target
名,同时还需要把生成的target
按照上层目录名来分组,例如下列目录结构:
examples
|- base
|- string
|- CMakeLists.txt
|- base.h
|- base.cpp
|- main.cpp
|- core
|- CMakeLists.txt
|- main.cpp
|- network
|- tcp
|- CMakeLists.txt
|- main.cpp
这个项目需要生成Visual Studio项目,CMakeLists.txt
如下所示:
# remove last end of "/"
string(REGEX REPLACE "/$" "" CURRENT_FOLDER_ABSOLUTE ${CMAKE_CURRENT_SOURCE_DIR})
# get current relative dir name and set target name
string(REGEX REPLACE ".*/(.*)" "\\1" CURRENT_FOLDER ${CURRENT_FOLDER_ABSOLUTE})
set (TARGET_NAME ${CURRENT_FOLDER})
# init target
add_executable (${TARGET_NAME} ${ARGN})
target_include_directories (${TARGET_NAME} ${CMAKE_CURRENT_SOURCE_DIR})
# get above dir name and set target group name
get_filename_component(SECOND_FOLDER_ABSOLUTE ${CURRENT_FOLDER_ABSOLUTE} DIRECTORY)
string(REGEX REPLACE ".*/(.*)" "\\1" SECOND_FOLDER ${SECOND_FOLDER_ABSOLUTE})
# group target
set_target_properties (${TARGET_NAME} PROPERTIES FOLDER "examples/${SECOND_FOLDER}")
生成后的项目如图所示,这里借用一个项目结构相同的项目来展示。
进入正题,获取相对路径的目录名有两种方式:
-
利用字符串正则表达式替换
string(REGEX REPLACE <regular_expression> <replacement_expression> <output_variable> <input> [<input>...])
-
利用命令
get_filename_component
不管哪种,都要先去掉绝对路径后面的/
。
正则表达式
string(REGEX REPLACE ".*/(.*)" "\\1" CURRENT_FOLDER ${CURRENT_FOLDER_ABSOLUTE})
使用括号来保存子表达式的匹配结果,子表达式匹配的就是相对路径的目录名。这种用法可以参考google protobuf的写法。
获取上层目录也是简单的,只需要使用正则表达式替换掉当前目录名,然后再执行一次获取当前目录名即可。
string(REGEX REPLACE "(.*)/${CURRENT_FOLDER}$" "\\1" SECOND_FOLDER_ABSOLUTE ${CURRENT_FOLDER_ABSOLUTE})
string(REGEX REPLACE ".*/(.*)" "\\1" SECOND_FOLDER ${SECOND_FOLDER_ABSOLUTE})
这里要清楚正则表达式之中的括号子表达式和"\\1"
的含义就行了Search and Replace With Regular Expressions,千变万化都可以。
get_filename_component
这个命令就更好理解了,把目录名当成文件名就行了;
获取当前目录名:
get_filename_component(CURRENT_FOLDER ${CURRENT_FOLDER_ABSOLUTE} NAME)
获取上层目录名:
get_filename_component(SECOND_FOLDER_ABSOLUTE ${CURRENT_FOLDER_ABSOLUTE} DIRECTORY)
get_filename_component(SECOND_FOLDER ${SECOND_FOLDER_ABSOLUTE} NAME)