• C++-CMake指令:include指令【.cmake文件/MACRO宏/function函数】


    说到cmake,可能最先想到的就是CmakeLists.txt文件,但是在很多情况下,也会看到.cmake文件。也许,你会诧异,.cmake文件是干什么的,甚至会想.cmake文件是不是cmake的正统文件,而CmakeLists.txt并不是。

    但其实,CmakeLists.txt才是cmake的正统文件,而.cmake文件是一个模块文件,可以被include到CMakeLists.txt中。
     

    一、include指令

    include指令一般用于语句的复用,也就是说,如果有一些语句需要在很多CMakeLists.txt文件中使用,为避免重复编写,可以将其写在.cmake文件中,然后在需要的CMakeLists.txt文件中进行include操作就行了

    include指令的结构为:

    1. include(<file|module> [OPTIONAL] [RESULT_VARIABLE <var>]
    2. [NO_POLICY_SCOPE])

    虽然,有不少的可选参数,但是一般情况下,都是直接写:

    include(file|module)
    

    注意,为了使CMakeLists.txt能够找到该文件,需要指定文件完整路径(绝对路径或相对路径),当然如果指定了CMAKE_MODULE_PATH,就可以直接include该目录下的.cmake文件了。

    .cmake文件里面通常是什么信息呢?

    .cmake文件里包含了一些cmake命令和一些宏/函数,当CMakeLists.txt包含该.cmake文件时,当编译运行时,该.cmake里的一些命令就会在该包含处得到执行,并且在包含以后的地方能够调用该.cmake里的一些函数

    什么是宏?什么是函数?

    二、MACRO宏和function函数

    1、宏和函数的定义

    先看一下关键字:cmake的宏是MACRO,函数是function。它们的用法是:

    1. macro( [arg1 [arg2 [arg3 ...]]])
    2. COMMAND1(ARGS ...) # 命令语句
    3. COMMAND2(ARGS ...)
    4. ...
    5. endmacro()
    6. function( [arg1 [arg2 [arg3 ...]]])
    7. COMMAND1(ARGS ...) # 命令语句
    8. COMMAND2(ARGS ...)
    9. ...
    10. function()

    定义一个名称为name的宏(函数),arg1...是传入的参数。我们除了可以用${arg1}来引用变量以外,系统为我们提供了一些特殊的变量:

    变量说明
    argv##是一个下标,0指向第一个参数,累加
    argv所有的定义时要求传入的参数
    argn定义时要求传入的参数以外的参数
    argc传入的实际参数的个数,也就是调用函数是传入的参数个数

    宏和函数的区别

    那么宏和函数之间的区别是什么呢?

    其实和C/C++里面宏和函数之间的区别差不多,宏就是字符串替换,函数就是使用变量,在命令中途可以对改变量进行修改

    StackOverflow的例子来了解一下区别:

    首先创建一个CMakeLists.txt

    1. cmake_minimum_required(VERSION 3.0)
    2. include(test.cmake)

    在同目录下创建文件test.cmake

    1. set(var "ABC")
    2. macro(Moo arg)
    3. message("arg = ${arg}")
    4. set(arg "abc")
    5. message("# After change the value of arg.")
    6. message("arg = ${arg}")
    7. endmacro()
    8. message("=== Call macro ===")
    9. Moo(${var})
    10. function(Foo arg)
    11. message("arg = ${arg}")
    12. set(arg "abc")
    13. message("# After change the value of arg.")
    14. message("arg = ${arg}")
    15. endfunction()
    16. message("=== Call function ===")
    17. Foo(${var})

    运行cmake:

    1. mkdir build && cd build
    2. cmake ..

    运行后的输出结果是:

    1. -- The C compiler identification is GNU 5.4.0
    2. -- The CXX compiler identification is GNU 5.4.0
    3. -- Check for working C compiler: /usr/bin/cc
    4. -- Check for working C compiler: /usr/bin/cc -- works
    5. -- Detecting C compiler ABI info
    6. -- Detecting C compiler ABI info - done
    7. -- Detecting C compile features
    8. -- Detecting C compile features - done
    9. -- Check for working CXX compiler: /usr/bin/c++
    10. -- Check for working CXX compiler: /usr/bin/c++ -- works
    11. -- Detecting CXX compiler ABI info
    12. -- Detecting CXX compiler ABI info - done
    13. -- Detecting CXX compile features
    14. -- Detecting CXX compile features - done
    15. === Call macro ===
    16. arg = ABC
    17. # After change the value of arg.
    18. arg = ABC
    19. === Call function ===
    20. arg = ABC
    21. # After change the value of arg.
    22. arg = abc
    23. -- Configuring done
    24. -- Generating done
    25. -- Build files have been written to: /home/yngzmiao/test/build

    从这里可以看出,宏实现的仅仅是字符串替换,宏定义的过程中是无法进行修改的,而函数却是可以的

    蛋疼的参数

    一般情况下,从上面的例子就能看出宏和函数的用法了,但很多情况下,我们自以为的“懂了”都是假懂。比如一不小心,就会出错。

    更换test.cmake为下面的内容,并运行:

    1. set(var "ABC")
    2. macro(Moo arg)
    3. message("arg = ${arg}")
    4. set(arg "abc")
    5. message("# After change the value of arg.")
    6. message("arg = ${arg}")
    7. endmacro()
    8. message("=== Call macro ===")
    9. Moo(var)
    10. function(Foo arg)
    11. message("arg = ${arg}")
    12. set(arg "abc")
    13. message("# After change the value of arg.")
    14. message("arg = ${arg}")
    15. endfunction()
    16. message("=== Call function ===")
    17. Foo(var)

    运行后的输出结果是:

    1. -- The C compiler identification is GNU 5.4.0
    2. -- The CXX compiler identification is GNU 5.4.0
    3. -- Check for working C compiler: /usr/bin/cc
    4. -- Check for working C compiler: /usr/bin/cc -- works
    5. -- Detecting C compiler ABI info
    6. -- Detecting C compiler ABI info - done
    7. -- Detecting C compile features
    8. -- Detecting C compile features - done
    9. -- Check for working CXX compiler: /usr/bin/c++
    10. -- Check for working CXX compiler: /usr/bin/c++ -- works
    11. -- Detecting CXX compiler ABI info
    12. -- Detecting CXX compiler ABI info - done
    13. -- Detecting CXX compile features
    14. -- Detecting CXX compile features - done
    15. === Call macro ===
    16. arg = var
    17. # After change the value of arg.
    18. arg = var
    19. === Call function ===
    20. arg = var
    21. # After change the value of arg.
    22. arg = abc
    23. -- Configuring done
    24. -- Generating done
    25. -- Build files have been written to: /home/yngzmiao/test/build

    对比两段程序可以看出其中的区别:无论是宏还是函数,当调用的时候如果使用的是set出来的变量,都必须通过${}将变量的内容传递进去,而不能只写上变量名

    这是将实参传递给形参时的注意点,但在宏和函数的实现过程中,还有需要注意的内容。

    例子:

    1. set(var "ABC")
    2. macro(Moo arg)
    3. if (arg STREQUAL "ABC")
    4. message("arg1 = ${arg}")
    5. endif()
    6. if (${arg} STREQUAL "ABC")
    7. message("arg2 = ${arg}")
    8. endif()
    9. endmacro()
    10. message("=== Call macro ===")
    11. Moo(${var})
    12. function(Foo arg)
    13. if (arg STREQUAL "ABC")
    14. message("arg1 = ${arg}")
    15. endif()
    16. if (${arg} STREQUAL "ABC")
    17. message("arg2 = ${arg}")
    18. endif()
    19. endfunction()
    20. message("=== Call function ===")
    21. Foo(${var})

    运行后的输出结果是:

    1. -- The C compiler identification is GNU 5.4.0
    2. -- The CXX compiler identification is GNU 5.4.0
    3. -- Check for working C compiler: /usr/bin/cc
    4. -- Check for working C compiler: /usr/bin/cc -- works
    5. -- Detecting C compiler ABI info
    6. -- Detecting C compiler ABI info - done
    7. -- Detecting C compile features
    8. -- Detecting C compile features - done
    9. -- Check for working CXX compiler: /usr/bin/c++
    10. -- Check for working CXX compiler: /usr/bin/c++ -- works
    11. -- Detecting CXX compiler ABI info
    12. -- Detecting CXX compiler ABI info - done
    13. -- Detecting CXX compile features
    14. -- Detecting CXX compile features - done
    15. === Call macro ===
    16. arg2 = ABC
    17. === Call function ===
    18. arg1 = ABC
    19. arg2 = ABC
    20. -- Configuring done
    21. -- Generating done
    22. -- Build files have been written to: /home/yngzmiao/test/build

    可以看出,在宏和函数的实现过程中,宏的参数由于不是传统意义上的变量,而是字符串替换,因此需要通过${}取出内容。而函数却不一定需要这样

    也就是说,对于macro宏而言:

    1. if(argv0) # 错误用法
    2. if(${argv0}) # 正确用法
    3. if(defined argv0) # 错误用法
    4. if(defined ${argv0}) # 正确用法

    也就是说,对于宏和函数的参数而言:

    • 当宏和函数调用的时候,如果传递的是经set设置的变量,必须通过${}取出内容
    • 在宏的定义过程中,对变量进行的操作必须通过${}取出内容,而函数就没有这个必要

    【CMake】cmake中的include指令(.cmake文件/MACRO宏/function函数)_Yngz_Miao的博客-CSDN博客

    CMake中include指令介绍_liitdar的博客-CSDN博客_cmake include 

    cmake(三十五)Cmake之include指令_wzj_110的博客-CSDN博客 

    CMake中include指令介绍_liitdar的博客-CSDN博客_cmake include 

  • 相关阅读:
    Qt应用软件【文件篇】json文件读写
    手把手教你搭建规范的团队vue项目,包含commitlint,eslint,prettier,husky,commitizen等等
    Yalmip+DCOPF+直流最优潮流
    【浙政钉】第一篇:企业内应用免登
    [编程题]数据库连接池 - 牛客网题解
    Java 数组_方法_static关键字
    在linux系统中使用WoeUSB工具制作U盘启动盘
    python进制和编码
    关于系统架构
    Kafka数据消费可靠性分析_java培训
  • 原文地址:https://blog.csdn.net/u013250861/article/details/127935572