在学习 cmake 之前,首先需要了解 cmake 的约定和基础语法
基本约定
cmake 的文件名为
CMakeLists.txt。该文件名大小写敏感,且不能更改cmake 的语句应当为小写
cmake 自带的宏应当全大写
自行设置的变量名应当小写
数据类型、循环和条件语句
cmake 中的行注释为 # ,其用法与 C++ 中的 // 完全相同
使用 message(变量名或字符串) 可以向控制台输出信息 |
数据类型
对于 cmake 来说,数据类型只有三种: 字符串 、 布尔值 和 列表。 要设置一个变量,可以通过 set(变量名 …) 的方式,要访问一个变量,可以直接使用 ${变量名} 的方式
使用 $变量名 也可以访问变量,但是在拼接字符串时会导致解析变量名错误,因此极度不推荐。 |
字符串
要设置一个 字符串 ,使用 set(变量名 字符串) 的方式。
cmake 中的字符串无需加单引号或双引号(也不能加)
要拼接字符串,可以直接加到变量名后,例如:
set(var1 hello) message(${var1}world) # 输出结果为 helloworld字符串中间的空格将被忽略。例如:
set(var1 sd) # 相当于 C++ 中的 ``string var1 = "sd"``
布尔值
要设置一个 布尔值 , 使用 set(变量名 布尔值) 的方式。
以下布尔值为真:1、ON、YES、Y、TURE。
以下布尔值为假:0、OFF、NO、N、FALSE、IGNORE、NOTFOUND、空值、以
-NOTFOUND为后缀的变量名
|
列表
要设置一个 列表 , 使用 list(操作方式 变量名 字符串) 的方式。 例如:
list(APPEND var ss ss)使用 set(变量名 值1 值2 值3) 得到的也是列表 |
条件语句
cmake 使用 if() elseif() else() endif() 控制整个条件语句
例如:
if(true)
message(true)
elseif(false)
message(false)
endif()| else() 和 endif() 用于指定指令的作用域 |
循环
cmake有两种控制循环的方式:
foreach endforeach
while endwhile
使用 break 可以跳出循环
例如:
list(APPEND var h e w o r d)
foreach(_li ${var})
message(${_li})
endforeach()- 输出结果为
h e w o r d
正确区分 Clang
Clang 的 API 有两种模式:
clang 模拟 GNU API
clang-cl 模拟 MSVC API
但无论是哪一种,都会导致 CMAKE_CXX_COMPILER_ID 被设置为 Clang。正确的方式为:
# Get compiler info
set(CXX_FLAGS_STYLE_GNU OFF)
set(CXX_FLAGS_STYLE_MSVC OFF)
set(CXX_FLAGS_STYLE_CLANGCL OFF)
if ("${CMAKE_CXX_COMPILER_ID}" STREQUAL"GNU")
set(CXX_FLAGS_STYLE_GNU ON)
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL"Clang")
if ("${CMAKE_CXX_SIMULATE_ID}" STREQUAL"MSVC")
set(CXX_FLAGS_STYLE_CLANGCL ON)
else ()
set(CXX_FLAGS_STYLE_GNU ON)
endif ()
elseif ("${CMAKE_CXX_COMPILER_ID}" STREQUAL"MSVC")
set(CXX_FLAGS_STYLE_MSVC ON)
else ()
message(FATAL_ERROR"Unsupported compiler!")
endif ()
# Compiler flags
target_compile_options(${PROJECT_NAME}
PRIVATE
## GCC/Clang Flags
$<$<BOOL:${CXX_FLAGS_STYLE_GNU}>:"-someflag">
## MSVC flags
$<$<BOOL:${CXX_FLAGS_STYLE_MSVC}>:"/someflag">
## CLANG-CL flags
$<$<BOOL:${CXX_FLAGS_STYLE_CLANGCL}>:"-someflag">
)实用技巧
启用 strip
add_link_options($<$<CONFIG:MinSizeRel>:-s>)