cmake函数参数解析
来源:互联网 发布:sql join where 区别 编辑:程序博客网 时间:2024/06/10 22:43
最近在迁移公司的make系统到cmake上,发现cmake的function参数很奇怪。例如,如果我们向一个function传递list作为参数,在function中,形参会变成如下状况:
set(SRC)list(APPEND SRC a.cpp b.cpp)list(APPEND SRC c.cpp d.cpp)function(tst_arguments src_list)message("src_list = "${src_list})endfunction()message("SRC = "${SRC})tst_arguments(${SRC})==== output ====SRC = a.cppb.cppc.cppd.cppsrc_list = a.cpp很奇怪的是,这里的${SRC}在function外是完整的4个元素,而在function却只剩下了头一个元素(可能跟list的定长有关)。如果我们要传给function以n个源文件组成的list,这样显然不行。
一种简单的解决方法是使用ARGV,ARGC配合,他们的含义如同C/C++中main的argv和argc,分别代表参数和参数个数,使用如下方法解析参数:
function(tst_arguments src_list)message("ARGC = "${ARGC})message("ARGV = "${ARGV}) set(INDEX 0) while(INDEX LESS ${ARGC})message("ARG = "${ARGV${INDEX}})math(EXPR INDEX "${INDEX} + 1")endwhile()endfunction()tst_arguments(${SRC})==== output ====ARGC = 4ARGV = a.cppb.cppc.cppd.cppARG = a.cppARG = b.cppARG = c.cppARG = d.cpp当然,你也可以使用cmake的foreach循环遍历参数。这招对付只有一个list的参数时十分有效,但是在出现多个参数的情况就很麻烦,如下:
##假设函数link_lib将src_list中的源文件链接成库,根据type制定是链接静态库还是动态库#function(link_lib src_list type) message("ARGC = "${ARGC})message("ARGV = "${ARGV}) #以下根据参数的实际情做了操作,手动处理,以保证正确获取src_list和type set(INDEX 0) math(EXPR MAX "${ARGC} - 1") while(INDEX LESS ${MAX}) #do something to link math(EXPR INDEX "${INDEX} + 1") endwhile()endfunction()link_lib(${SRC} , so)==== output ====ARGC = 5ARGV = a.cppb.cppc.cppd.cppso原来,ARG把两个参数混在了一起,虽然后面我们使用while进行了特殊处理,但是这对于cmake的函数不具备普遍性,移植起来很麻烦。
决定版的solution是使用cmake的cmake_parse_arguments来解析函数参数,它有点像解析一个map键值对,首先看下它的函数原型:
include (CMakeParseArguments) #必须包含这个cmake文件才能使用<span class="highlighted">cmake_parse_arguments</span>CMAKE_PARSE_ARGUMENTS(<prefix> <options> <one_value_keywords> <multi_value_keywords> args...)首先,prefix是一个前缀,等会儿在引用参数的时候会提到,<option>是一个列表,里面可以包含一些你感兴趣的KeyWord,随后可以通过它来看看你所需要的KeyWord是否被设置。<one_value_keywords>是一个单值参数的KeyWord列表。<multi_value_keywords>是一个多值参数的KeyWord列表(如list),下面举个例子,看看如何使用它们,首先定义所需要的函数,由于参数是由CMAKE_PARSE_ARGUMENTS来解析的,所以在函数声明中就不需要定义参数了:
function(tst_arguments) CMAKE_PARSE_ARGUMENTS( TEST "" "NAME;COMMAND;BASELINE" "ARGSLIST" ${ARGN} ) message("TEST_DEFAULT_ARGS is ${TEST_DEFAULT_ARGS} from ${ARGN}") message("TEST_NAME is ${TEST_NAME}") message("TEST_COMMAND is ${TEST_COMMAND}") message("TEST_ARGSLIST is ${TEST_ARGSLIST}") message("TEST_BASELINE is ${TEST_BASELINE}")endfunction(tst_arguments)这里的前缀是TEST,<one_value_keywords>我们设置单值参数的KeyWord(NAME;COMMAND;BASELINE),这将在随后的函数调用中注明KeyWord和Value的关系,<multi_value_keywords>我们设置多值参数的KeyWord("ARGSLIST"),调用函数:
TEST_ARGUMENT( NAME testiso COMMAND "RunMe" ARGSLIST ${SRC} BASELINE "/home/sakaue/iWork")==== output ====TEST_DEFAULT_ARGS is from NAME;testiso;COMMAND;RunMe;ARGSLIST;a.cpp;b.cpp;c.cpp;d.cpp;BASELINE;/home/sakaue/iWorkTEST_NAME is testisoTEST_COMMAND is RunMeTEST_ARGSLIST is a.cpp;b.cpp;c.cpp;d.cppTEST_BASELINE is /home/sakaue/iWork可以看见,这里调用时的参数传递如同map一样<NAME ,testiso_${datafile} >,<COMMAND , "RunMe">,<ARGSLIST , ${SRC}>等等,在函数中,使用 前缀+KeyWord 来调用Value,这样比自己解析参数方便许多,而且也不会在还有list参数时和其他类型函数混在一起的情况。
更多讯息参考:http://www.cmake.org/cmake/help/v3.0/module/CMakeParseArguments.html?highlight=cmake_parse_arguments
4 0
- cmake函数参数解析
- cmake函数参数解析
- cmake函数参数解析
- main函数参数解析
- main函数参数解析
- main函数参数解析
- main函数参数解析
- 命令行参数解析函数getopt_long()
- 命令行参数解析函数--getopt
- matlab mvnrnd函数参数解析
- listen函数backlog参数解析
- main函数的参数解析
- mysql cmake参数详解
- cmake参数说明手册
- 学习解析命令行参数函数-getopt函数
- CMake 常用命令解析
- 函数可变参数列表解析,也就是函数参数不确定
- document.execCommand()函数可用参数解析
- LeetCode: Max Points on a Line
- 树链剖分-链的剖分(线段树维护+离线操作)
- 关于ios越狱开发的那些事
- 7518爱的是看了就关联方看见过开了房7518季后赛的规范健康
- 微软MSDN订阅介绍
- cmake函数参数解析
- VC 调用 VB 的DLL
- 老罗锤子手机发布会,我感到深深地愧疚!
- uva 10453 - Make Palindrome
- 用iOSOpenDev在Xcode里配置越狱开发环境
- KeyListener的三个方法,而其参数KeyEvent却不同
- <转载>Android软件开发之获取通讯录联系人信息
- Map集合及使用
- hihoCoder:Trie图