1.4 用条件句控制编译
NOTE:这个示例代码可以在 https://github.com/dev-cafe/cmake-cookbook/tree/v1.0/chapter-01/recipe-04 找到,其中有一个C++示例。该配置在CMake 3.5版(或更高版本)测试有效的,并且已经在GNU/Linux、macOS和Windows上进行了测试。
目前为止,看到的示例比较简单,CMake执行流是线性的:从一组源文件到单个可执行文件,也可以生成静态库或动态库。为了确保完全控制构建项目、配置、编译和链接所涉及的所有步骤的执行流,CMake提供了自己的语言。本节中,我们将探索条件结构if-else- else-endif
的使用。
NOTE: CMake语言相当庞杂,由基本的控制结构、特定于CMake的命令和使用新函数模块化扩展语言的基础设施组成。完整的概览可以在这里找到: https://cmake.org/cmake/help/latest/manual/cmake-language.7.html
具体实施
从与上一个示例的的源代码开始,我们希望能够在不同的两种行为之间进行切换:
- 将
Message.hpp
和Message.cpp
构建成一个库(静态或动态),然后将生成库链接到hello-world
可执行文件中。 - 将
Message.hpp
,Message.cpp
和hello-world.cpp
构建成一个可执行文件,但不生成任何一个库。
让我们来看看如何使用CMakeLists.txt
来实现:
首先,定义最低CMake版本、项目名称和支持的语言:
cmake_minimum_required(VERSION 3.5 FATAL_ERROR)
project(recipe-04 LANGUAGES CXX)
我们引入了一个新变量
USE_LIBRARY
,这是一个逻辑变量,值为OFF
。我们还打印了它的值:set(USE_LIBRARY OFF)
message(STATUS "Compile sources into a library? ${USE_LIBRARY}")
CMake中定义
BUILD_SHARED_LIBS
全局变量,并设置为OFF
。调用add_library
并省略第二个参数,将构建一个静态库:set(BUILD_SHARED_LIBS OFF)
然后,引入一个变量
_sources
,包括Message.hpp
和Message.cpp
:list(APPEND _sources Message.hpp Message.cpp)
然后,引入一个基于
USE_LIBRARY
值的if-else
语句。如果逻辑为真,则Message.hpp
和Message.cpp
将打包成一个库:if(USE_LIBRARY)
# add_library will create a static library
# since BUILD_SHARED_LIBS is OFF
add_library(message ${_sources})
add_executable(hello-world hello-world.cpp)
target_link_libraries(hello-world message)
else()
add_executable(hello-world hello-world.cpp ${_sources})
endif()
我们可以再次使用相同的命令集进行构建。由于
USE_LIBRARY
为OFF
,hello-world
可执行文件将使用所有源文件来编译。可以通过在GNU/Linux上,运行objdump -x
命令进行验证。
工作原理
我们介绍了两个变量:USE_LIBRARY
和BUILD_SHARED_LIBS
。这两个变量都设置为OFF
。如CMake语言文档中描述,逻辑真或假可以用多种方式表示:
- 如果将逻辑变量设置为以下任意一种:
1
、ON
、YES
、true
、Y
或非零数,则逻辑变量为true
。 - 如果将逻辑变量设置为以下任意一种:
0
、OFF
、NO
、false
、N
、IGNORE、NOTFOUND
、空字符串,或者以-NOTFOUND
为后缀,则逻辑变量为false
。
USE_LIBRARY
变量将在第一个和第二个行为之间切换。BUILD_SHARED_LIBS
是CMake的一个全局标志。因为CMake内部要查询BUILD_SHARED_LIBS
全局变量,所以add_library
命令可以在不传递STATIC/SHARED/OBJECT
参数的情况下调用;如果为false
或未定义,将生成一个静态库。
这个例子说明,可以引入条件来控制CMake中的执行流。但是,当前的设置不允许从外部切换,不需要手动修改CMakeLists.txt
。原则上,我们希望能够向用户开放所有设置,这样就可以在不修改构建代码的情况下调整配置,稍后将展示如何做到这一点。
NOTE:else()
和endif()
中的()
,可能会让刚开始学习CMake代码的同学感到惊讶。其历史原因是,因为其能够指出指令的作用范围。例如,可以使用if(USE_LIBRARY)…else(USE_LIBRARY)…endif(USE_LIBIRAY)
。这个格式并不唯一,可以根据个人喜好来决定使用哪种格式。
TIPS:_sources
变量是一个局部变量,不应该在当前范围之外使用,可以在名称前加下划线。