CMake学习笔记(六)可以在CMake的function中修改function函数体之外的变量值吗?
目录
问题由来
解决办法
方法1 set函数结合PARENT_SCOPE
代码
效果
方法2 使用PARENT_SCOPE结合双重${}
代码
效果
方法3 使用CACHE
代码
效果
问题由来
比如说,CMakeLists.txt所在路径下还有2个文件夹,且每个文件夹下各自有两个文件:一个cpp,一个h。
如何使用CMake的一个function遍历这两个文件,把里面的内容打印出来?
解决办法
以下三个办法都受了deepseek解答的启发,亲测有效
方法1 set函数结合PARENT_SCOPE
代码
cmake_minimum_required(VERSION 3.22.1)project(parent)function (addpath)file(GLOB var1 "view/*.cpp")file(GLOB var2 "model/*.cpp")string(APPEND SRC_CPP "${var1}")#注意,这里一定要用string,不能用setstring(APPEND SRC_CPP "${var2}")set(SRC_CPP "${SRC_CPP}" PARENT_SCOPE)file(GLOB var1 "view/*.h")file(GLOB var2 "model/*.h")string(APPEND SRC_HDR "${var1}")#注意,这里一定要用string,不能用setstring(APPEND SRC_HDR "${var2}")set(SRC_HDR "${SRC_HDR}" PARENT_SCOPE)
endfunction()addpath()message(STATUS "SRC_HDR=${SRC_HDR}\nSRC_CPP=${SRC_CPP}")
file(GLOB 变量名 "正则表达式")函数从正则表达式中找到所有匹配的文件(这里是子目录view或者model中的h文件和cpp文件),获取所有头文件和cpp文件的路径,然后将路径输入到 变量var1 和var2中。
string(APPEND ...)将var1 和 var2的内容添加到SRC_CPP或者SRC_HDR中。注意,string这里操作的变量的作用域都在function内部。
set(....PARENT_SCOPE)将SRC_HDR/SRC_CPP的内容传递到function作用域的上一级。
效果
方法2 使用PARENT_SCOPE结合双重${}
代码
cmake_minimum_required(VERSION 3.22.1)project(parent)set(CPP_SRC "")
set(CPP_HDR "")function (addpath path suffix)file(GLOB var1 "view/*.${suffix}")file(GLOB var2 "model/*.${suffix}")set(var "${var1} ${var2}")set(${path} "${${path}} ${var}" PARENT_SCOPE)
endfunction()addpath(CPP_SRC "cpp")
addpath(CPP_HDR "h")message("STATUS" "CPP_HDR=${CPP_HDR} CPP_SRC=${CPP_SRC}")
1 这里语句file(GLOB var1 "view/*.${suffix}")使用了${}符号。 使用了这个符号之后,file指令不会直接使用suffix作为其输入变量,而是将suffix视为一个变量,采用变量的值作为输入变量。
比如addpath(CPP_SRC "cpp"),这里suffix = "cpp",所以file(GLOB var1 "view/*.${suffix}")会把cpp代入suffix,变为file(GLOB var1 "view/*.cpp")再执行。
2 同样的道理,set (${path} "${${path}} ${var}" PARENT_SCOPE)把${path}视为变量的内容。在addpath(CPP_SRC "cpp")中,path = CPP_SRC。所以set (${path} "${${path}} ${var}" PARENT_SCOPE)就变成了set(CPP_SRC "${CPP_SRC}..." PARENT_SCOPE)。而第二个CPP_SRC的外面因为又有一层${},所以${SRC_CPP}又指向了SRC_CPP的内容。由于CPP_SRC一开始被赋值为"",所以set (${path} "${${path}} ${var}" PARENT_SCOPE)最后变成了
set (SRC_CPP """ ${var}" PARENT_SCOPE)。也就是说,SRC_CPP=""+${var}=${var}
这个值var由前面set(var "${var1} ${var2}")决定。
3 最后,set (SRC_CPP """ ${var}" PARENT_SCOPE)由于使用了PARENT_SCOPE,所以其结果作用在上一级--也就是调用addpath这一层级--的变量SRC_CPP。
效果
方法3 使用CACHE
代码
cmake_minimum_required(VERSION 3.22.1)project(parent)function (addpath)file(GLOB var1 "view/*.cpp")file(GLOB var2 "model/*.cpp")set(SRC_CPP "${var1} ${var2}" CACHE STRING "cpp path" FORCE)file(GLOB var1 "view/*.h")file(GLOB var2 "model/*.h")set(SRC_HDR "${var1} ${var2}" CACHE STRING "hpp path" FORCE)
endfunction()addpath()message(STATUS "SRC_HDR=${SRC_HDR}\nSRC_CPP=${SRC_CPP}")
这里set函数与CACHE配合使用。 这样配合使用的结果是,set操作的变量(这里是SRC_CPP和SRC_HDR)被存入CMakeCache.txt文件:
//cpp path
SRC_CPP:STRING=C:/code/cmake/parntScope/view/view.cpp C:/code/cmake/parntScope/model/model.cpp//hpp path
SRC_HDR:STRING=C:/code/cmake/parntScope/view/view.h C:/code/cmake/parntScope/model/model.h
存入CMakeCache.txt之后, SRC_CPP和SRC_HDR就成了全局变量,作用域不仅限于function,而是整个CMake项目。甚至在CMake通过add_subdirectory包括了其他CMakeLists.txt的情况下,各级CMakeLists.txt都可以访问这些全局变量。
最后,set函数还加了FORCE选项。FORCE的作用如下:覆盖CMakeCache.txt中已有的赋值。假如CMakeCache.txt已经含有了SRC_CPP和SRC_HDR。那么仅仅调用set(....CACHE...)是不会再起作用的。要想覆盖CMakeCache.txt中已有的赋值,就要调用set(...CACHE..FORCE)。