CMake中macro的使用

Posted fengbingchun

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了CMake中macro的使用相关的知识,希望对你有一定的参考价值。

      在https://blog.csdn.net/fengbingchun/article/details/127144948 中介绍了CMake中function的使用,这里介绍下macro的使用,它与function有很多的相似性。

      macro的定义格式如下:后面可作为命令供调用

macro(<name> [<arg1> ...])
  <commands>
endmacro()

      其中name是macro的名字,参数为arg1,arg2等。与function一样,macro名称也不区分大小写,但强烈建议使用macro定义中声明的相同名称。通常,macro使用全小写名称。示例代码段如下:

message("1.与function一样,macro名称也不区分大小写,但始终建议使用macro定义中声明的相同名称")
macro(csdn_addr)
    message("csdn addr: https://blog.csdn.net/fengbingchun")
endmacro()

csdn_addr()
CSDN_ADDR()
csdn_ADDR()

      macro参数:与function一样,macro也采用两种类型的参数:
      (1).命名参数或关键字参数(named or keyword arguments):是必须的,如果未提供将触发error。参数名称之间不需要逗号。示例代码段如下:

message("2.命名参数是必须的,如果未提供将触发error.参数名称之间不需要逗号")
macro(addr csdn github)
    message("csdn addr: $csdn")
    message("github addr: $github")
endmacro()

addr("https://blog.csdn.net/fengbingchun" "https://github.com/fengbingchun")

      (2).可选参数(optional arguments):可以使用一些预定义的变量访问可选参数,示例代码段如下:
      ARGC:参数总数(命名参数+可选参数)。
      ARGV:包含命名参数和可选参数的变量列表
      ARGN:仅包含可选参数的变量列表

message("3.可以使用一些预定义的变量访问可选参数:ARGC, ARGV, ARGN")
macro(name_list name1 name2)
    message("argument count: $ARGC")
    message("all arguments: $ARGV")
    message("optional arguments: $ARGN")
endmacro()

name_list(Jack Kate Jony Tom)
# only named arguments
name_list(Jack Kate)

      除了这三个变量之外,cmake还提供了ARGV0,ARGV1,ARGV2,...,这将具有传入参数的实际值。引用ARGC之外的ARGV#参数将导致未定义的行为。注意:这些在function中起作用,在macro中直接使用无效。在function中,它们是CMake意义上的真变量;在macro中,它们不是,它们是字符串替换,就像C预处理器对macro所做的那样。

message("4.cmake还提供了ARGV0,ARGV1,ARGV2,...,这将具有传入参数的实际值.引用ARGC之外的ARGV#参数将导致未定义的行为.这些在function中起作用,在macro中不作改动无效")
macro(programming_language name1 name2)
    # 在某些情况下,macro在使用特殊变量时会表现出奇怪的行为,注意:此处与function的差异
    if($ARGV0) # 好像不起作用
        message("ARGV0: $ARGV0")
    else()
        message("ARGV0 not defined")
    endif()

    math(EXPR last_index "$ARGC-1")
    foreach(index RANGE 0 $last_index)
        # 在某些情况下,macro在使用特殊变量时会表现出奇怪的行为,注意:此处与function的差异
        message("argument at index $index: $ARGV$index")
    endforeach()

    # 通常,使用变量访问命名参数,使用ARGN访问可选参数
    message("name1: $name1")
    message("name2: $name2")

    # 注意:此处与function的差异
    set(list_var $ARGN)
    #message("list var: $list_var")
    #foreach(arg IN LISTS $ARGN) # 好像不起作用
    foreach(arg IN LISTS list_var)
        message("optional name: $arg")
    endforeach()
endmacro()

programming_language(C++ Python Go Matlab)

      通常,使用变量访问命名参数,使用ARGN访问可选参数

      使用DEFINED关键字,你可以检查是否定义了给定的变量、缓存变量或环境变量。变量的值无关紧要。在某些情况下,macro在使用特殊变量时会表现出奇怪的行为。

message("5.使用DEFINED关键字,你可以检查是否定义了给定的变量、缓存变量或环境变量.变量的值无关紧要.在function中起作用,在macro中无效")
macro(foo_macro name)
    # 在某些情况下,macro在使用特殊变量时会表现出奇怪的行为,注意:此处与function的差异
    if(DEFINED name)
        message("macro argument name is defined")
    else()
        message("macro argument name is not defined")
    endif()
endmacro()

function(foo_func name)
    if(DEFINED name)
        message("func argument name is defined")
    else()
        message("func argument name is not defined")
    endif()
endfunction()

foo_macro(csdn)
foo_func(csdn)

      变量作用域(variable scope):与function不同,macro不会引入新的作用域在macro中声明的变量(参数除外)将在调用后可用。function在调用时始终引入新作用域(new scope)。对于macro,macro调用将替换为marco主体(macro body),并使用字符串替换(string substitution)替换参数。因此,在某些情况下,function和macro的行为不同。

message("6.与function不同,macro不会引入新的作用域.在macro中声明的变量(参数除外)将在调用后可用")
set(development_language "C++")

macro(set_development_language name)
    message("macro param: $name")
    message("macro name: $development_language")
    set(new_language "Python")
    set(development_language "Matlab")
    message("macro new language: $new_language")
endmacro()

set_development_language("Go")
message("development_language: $development_language")
message("new language: $new_language")

      return()命令:由于macro不会创建任何新作用域,因此调用return()将退出当前作用域

message("7.由于macro不会创建任何新作用域,因此调用return()将退出当前作用域,这里是退出当前的.cmake文件,注意与在function里的差异:退出当前的function")
macro(early_return)
    message("csdn")
    return()
    message("github") # it will not be printed
endmacro()

early_return() # 退出当前cmake文件
message("will never be executed")

      如果定义了一个已经存在的macro时,将覆盖上一个macro。可以使用"_"加macro name的方式访问上一个macro。如果再次重新定义同一个macro,"_"加macro name版本将调用先前定义的macro。原始macro将永远不可用。如果开发人员在编写macro时不知道该macro已用于某些cmake库macro,则原始macro有可能永远隐藏。

message("8.如果定义了一个已经存在的macro时,将覆盖上一个macro.可以使用\\"_\\"加macro name的方式访问上一个macro.如果再次重新定义同一个macro,\\"_\\"加macro name版本将调用先前定义的macro.原始macro将永远不可用.")
macro(csdn_addr)
    message("csdn addr: https://github.com/fengbingchun")
endmacro()

csdn_addr() # csdn addr: https://github.com/fengbingchun
_csdn_addr() # csdn addr: https://blog.csdn.net/fengbingchun

macro(csdn_addr)
    message("csdn addr: https://www.baidu.com/")
endmacro()

csdn_addr() # csdn addr: https://www.baidu.com/
_csdn_addr() # csdn addr: https://github.com/fengbingchun

      也可通过cmake的预定义的命令cmake_parse_arguments来解析macro的参数。

      function和macro的不同
      (1).变量作用域不同:function会引入新的作用域,而macro不会,即,默认,function内新增的变量外部不可见;而macro内新增的变量外部可见。
      (2).cmake提供的变量ARGV0,ARGV1,ARGV2,...,在function中有效,在macro中无效。在某些情况下,macro中的ARGN也不会生效,在macro中,它们是字符串替换,就像C预处理器对macro所做的那样。
      (3).function内调用return()命令是退出当前function;而macro内调用return()是退出当前cmake文件。建议在macro中完全避免使用return()。
      (4).CMAKE_CURRENT_FUNCTION, CMAKE_CURRENT_FUNCTION_LIST_DIR, CMAKE_CURRENT_FUNCTION_LIST_FILE, CMAKE_CURRENT_FUNCTION_LIST_LIN这些变量应用于function中而不是macro中。

      测试代码组织结构如下:

      build.sh内容如下:

#! /bin/bash

# supported input parameters
params=(function macro)

usage()

	echo "Error: $0 needs to have an input parameter"

	echo "supported input parameters:"
	for param in $params[@]; do
		echo "  $0 $param"
	done

	exit -1


if [ $# != 1 ]; then
	usage
fi

flag=0
for param in $params[@]; do
	if [ $1 == $param ]; then
		flag=1
		break
	fi
done

if [ $flag == 0 ]; then
	echo "Error: parameter \\"$1\\" is not supported"
	usage
	exit -1
fi

if [[ ! -d "build" ]]; then
	mkdir build
	cd build
else
	cd build
fi

echo "==== test $1 ===="
cmake -DTEST_CMAKE_FEATURE=$1 ..

      CMakeLists.txt内容如下:

CMAKE_MINIMUM_REQUIRED(VERSION 3.13)
PROJECT(cmake_feature_usage)

if(TEST_CMAKE_FEATURE STREQUAL "function")
	include(test_function.cmake)
elseif(TEST_CMAKE_FEATURE STREQUAL "macro")
	include(test_macro.cmake)
endif()

message("==== test finish ====")

     test_macro.cmake内容:为上面所示代码段

     执行结果如下图所示:

      GitHubhttps://github.com/fengbingchun/Linux_Code_Test

以上是关于CMake中macro的使用的主要内容,如果未能解决你的问题,请参考以下文章

CMake中cmake_parse_arguments的使用

CMake中cmake_parse_arguments的使用

为啥在 CMake 配置和生成完成后使用 CLion 构建 OpenCV 时会出现错误?

使用CMake组织C++工程3:CMake 函数和宏

使用CMake检查列表是否包含特定条目的最佳方法

在宏内使用 Qt4 和 CMake 的 find_package 时出现问题