Hello系列 | cmake必备简明基础知识
liuian 2025-01-21 19:28 27 浏览
大家好,我是LinuxZn。
Hello系列,汇总短而实用的内容。
上一篇文章中我们分享了关于make与Makefile的知识:Hello系列 | Makefile必备基础知识梳理。make工具有很多种:gnu make、QT的qmake、微软的MS nmake等。不同的make工具遵循不同的规范,如果我们的程序想要运行在不同的平台上,就需要根据不同地平台的make工具规范编写对应的Makefile文件。显然,这很不方便。
CMake就是一个可以解决上面这个问题的工具。
什么是cmake?
CMake 是一个跨平台的安装(编译)工具。CMakeList.txt是一个与平台无关的、用于定制编译流程的文件。CMake 靠的是 CMakeLists.txt 文件来生成Makefile文件。
CMakeLists.txt文件的编写也需要遵循一些语法规则,CMakeLists.txt文件的语法与shell脚本的语法很相似,shell编程知识可见往期文章:Hello系列 | shell编程必备简明基础知识。
下面简单了解CMakeLists.txt简单的规则及一些示例。
cmake语法知识
1、直译模式
直译模式简单解释就是不生成Makefile的模式。这很方便我们验证一些CMakeLists.txt的语法及验证一些数学运算等。
下面通过简单实例区分直译模式与非直译模式的区别。
直译模式:
输入 -P参数 指定CMakeLists.txt脚本以直译模式解析。其中,message是CMakeLists.txt中用于输出信息的命令。以直译模式解析就不会生成Makefile文件,并且终端输出的信息就是我们CMakeLists.txt指定输出的内容。
非直译模式:
可见,以非直译模式解析则会生成Makefile文件,并且终端多输出了一些核查编译器相关的信息。
2、定义变量
CMakeLists.txt中只有字串和字串数组两种变量。定义变量通过 set命令 来定义,使用变量时在外面加上 ${} 符号即可。如:
# 定义变量
set(name "LinuxZn")
# 使用变量
message("My name is ${name}!")
① 注释使用符号 #。
② 命令不区分大小写,即set也可以替换为SET。
3、数学运算
# EXPR 是一款表达式计算工具
# math 是用于数学运算的命令
# 设置变量a、b的值
set(a "1")
set(b "2")
# 加
math(EXPR res "${a} + ${b}")
message("a + b : ${res}")
# 减
math(EXPR res "${a} - ${b}")
message("a - b : ${res}")
# 乘
math(EXPR res "${a} * ${b}")
message("a * b : ${res}")
# 除
math(EXPR res "${a} / ${b}")
message("a / b : ${res}")
# 取余
math(EXPR res "${a} % ${b}")
message("a % b : ${res}")
4、从命令行给变量赋值
# EXPR 是一款表达式计算工具
# math 是用于数学运算的命令
# 加
math(EXPR res "${a} + ${b}")
message("a + b : ${res}")
# 减
math(EXPR res "${a} - ${b}")
message("a - b : ${res}")
# 乘
math(EXPR res "${a} * ${b}")
message("a * b : ${res}")
# 除
math(EXPR res "${a} / ${b}")
message("a / b : ${res}")
# 取余
math(EXPR res "${a} % ${b}")
message("a % b : ${res}")
-D后面跟着变量及赋值。
我们经常会在命令行配置工程为debug模式还是release模式,如:
cmake -DCMAKE_BUILD_TYPE=Debug
CMAKE_BUILD_TYPE是cmake中的一个内置变量,用于指定构建类型。
5、流程控制
(1)if
set(ARCH "x86")
if(ARCH MATCHES "x86")
message("ARCH is x86")
else()
message("ARCH is arm")
endif()
(2)while
set(a "1")
while(${a} LESS "5")
message("${a}")
math(EXPR a "${a} + 1")
endwhile()
(3)foreach
message("for 1 =========")
foreach(i RANGE 1 5)
message("${i}")
endforeach()
message("for 2 =========")
foreach(i 1 5 6 7 9 10)
message("${i}")
endforeach()
message("for 3 =========")
foreach(str Linux C Cpp Python Shell)
message("${str}")
endforeach()
6、自定义宏与函数
(1)宏
# 定义名为printf的宏
macro(printf str)
message(${str})
endmacro()
# 使用
printf("hello macro")
(2)函数
# 定义名为printf的函数
function(printf str)
message(${str})
endfunction()
# 使用
printf("hello function")
(3)宏与函数的区别
函数中的变量是局部的,宏中的变量是全局的,宏中的变量在外面也可以被访问到。
# 定义名为func_printf的函数
function(func_printf str)
message(${str})
set(func_var "1111111111")
endfunction()
# 定义名为macro_printf的宏
macro(macro_printf str)
message(${str})
set(macro_var "222222222")
endmacro()
# 使用
func_printf("hello function")
message("func_var = ${func_var}")
macro_printf("hello macro")
message("macro_var = ${macro_var}")
7、查看cmake命令说明
上面列举的语法知识中,我们并未介绍所用命令的格式及使用方式。各命令详细的解释可以通过如下方式查看。
(1)查看所有cmake命令
cmake --help-command-list
(2)查看具体某个命令说明
比如,查看message命令说明:
cmake --help-command message
cmake与构建
上一节分享了cmake的一些基本语法知识。这一节我们一起来看一下cmake与构建相关的内容。
1、几个常用命令
下面列出几个常用的命令,在我们下面的例子中会用到。
(1)cmake_minimum_required
命令格式:
cmake_minimum_required(VERSION major.minor[.patch[.tweak]]
[FATAL_ERROR])
用于指定需要的 CMake 的最低版本。
使用示例:
cmake_minimum_required (VERSION 3.10)
(2)project
命令格式:
project(<PROJECT-NAME> [LANGUAGES] [<language-name>...])
用于指定项目的名称。
使用示例:
project (hello)
(3)add_executable
命令格式:
add_executable(<name> [WIN32] [MACOSX_BUNDLE]
[EXCLUDE_FROM_ALL]
source1 [source2 ...])
用于指定从一组源文件 source1 … 编译出一个可执行文件且命名为 name。
使用示例:
add_executable(hello main.c)
(4)aux_source_directory
命令格式:
aux_source_directory(<dir> <variable>)
用于将 dir 目录下的所有源文件的名字保存在变量 variable 中。
使用示例:
aux_source_directory(. DIR_SRCS)
(5)add_subdirectory
命令格式:
add_subdirectory(source_dir [binary_dir]
[EXCLUDE_FROM_ALL])
用于添加一个需要进行构建的子目录。
使用示例:
add_subdirectory(Lib)
(6)add_library
命令格式:
add_library(<name> INTERFACE [IMPORTED [GLOBAL]])
用于指定从一组源文件中编译出一个库文件且命名为name。
使用示例:
add_library(Lib ${DIR_SRCS})
(7)target_link_libraries
命令格式:
target_link_libraries(<target> ... <item>... ...)
用于指定 target 需要链接 item1 item2 …。
使用示例:
target_link_libraries(hello Lib)
(8)include_directories
命令格式:
include_directories([AFTER|BEFORE] [SYSTEM] dir1 [dir2 ...])
用于添加头文件路径。
使用示例:
include_directories(include)
2、常用内置变量
(1)PROJECT_SOURCE_DIR
目前正在处理中的专案最上层目录,即内含 project() 指令的 CMakeLists 所在资料夹。
(2)CMAKE_BUILD_TYPE
控制构建类型,可选值为:
- None: 编译器默认值
- Debug: 产生除错信息
- Release: 进行最佳化
- RelWithDebInfo: 进行最佳化,但仍然会启用 DEBUG flag
- MinSizeRel: 进行程式码最小化
(3)CMAKE_C_FLAGS
C编译器的编译选项。
(4)CMAKE_CXX_FLAGS
C++编译器的编译选项。
3、实例
(1)基础实例
main.c:
#include <stdio.h>
int main(void)
{
printf("hello cmake\n");
return 0;
}
CMakeLists.txt:
cmake_minimum_required (VERSION 3.10)
project (hello)
add_executable(hello main.c)
(2)多个文件、多个文件夹
上一个demo只有一个源文件,对应的CMakeLists.txt比较简单。下面看看有多个文件夹及文件的工程。
基于上面的demo,修改工程如:
main.c:
#include "hello.h"
int main(void)
{
print_hello();
return 0;
}
CMakeLists.txt:
cmake_minimum_required (VERSION 3.10)
project (hello)
# 添加头文件路径
include_directories(include)
# 查找src目录下的所有源文件并将名称保存到 SRC_DIR_SRCS 变量中
aux_source_directory(src SRC_DIR_SRCS)
# 查找当前目录下的所有源文件并将名称保存到 CUR_DIR_SRCS 变量中
aux_source_directory(. CUR_DIR_SRCS)
# 从SRC_DIR_SRCS与CUR_DIR_SRCS中编译出可执行文件hello
add_executable(hello
${SRC_DIR_SRCS}
${CUR_DIR_SRCS}
)
编译、运行:
cd build
cmake ..
make
./hello
(3)编译静态库
基于demo2,我们把src文件夹下的源文件编译成静态库,再由main.c调用。工程目录基本不变,需要在src下新增一个CMakeLists.txt文件,其内容如:
# 查找当前目录下的所有源文件并将名称保存到 DIR_LIB_SRCS 变量
aux_source_directory(. DIR_LIB_SRCS)
# 生成链接库
add_library (print_hello ${DIR_LIB_SRCS})
根目录下的CMakeLists.txt修改为:
cmake_minimum_required (VERSION 3.10)
project (hello)
# 添加头文件路径
include_directories(include)
# 查找当前目录下的所有源文件并将名称保存到 CUR_DIR_SRCS 变量中
aux_source_directory(. CUR_DIR_SRCS)
# 添加 src 子目录
add_subdirectory(src)
# 从CUR_DIR_SRCS中编译出可执行文件hello
add_executable(hello
${CUR_DIR_SRCS}
)
# 添加链接库
target_link_libraries(hello print_hello)
编译、运行:
cd build
cmake ..
make
./hello
(4)编译动态库
编译动态库的方式与编译动态库的方式差不多。基于上面的demo3,只需修改src文件夹下的CMakeLists.txt为:
# 查找当前目录下的所有源文件并将名称保存到 DIR_LIB_SRCS 变量
aux_source_directory(. DIR_LIB_SRCS)
# 生成动态库
add_library (print_hello SHARED ${DIR_LIB_SRCS})
编译、运行:
cd build
cmake ..
make
./hello
4、支持gdb调试
上面工程中根目录加上如下命令可支持gdb调试:
set(CMAKE_BUILD_TYPE "Debug")
set(CMAKE_CXX_FLAGS_DEBUG "$ENV{CXXFLAGS} -O0 -Wall -g -ggdb")
set(CMAKE_CXX_FLAGS_RELEASE "$ENV{CXXFLAGS} -O0 -Wall")
以上就是本次的分享。如果觉得文章有用,欢迎收藏、转发!
相关资料:
https://www.hahack.com/codes/cmake/
https://preshing.com/20170522/learn-cmakes-scripting-language-in-15-minutes/
https://cmake.org/
相关推荐
- 教你把多个视频合并成一个视频的方法
-
一.情况介绍当你有一个m3u8文件和一个目录,目录中有连续的视频片段,这些片段可以连成一段完整的视频。m3u8文件打开后像这样:m3u8文件,可以理解为播放列表,里面是播放视频片段的顺序。视频片段像这...
- 零代码编程:用kimichat合并一个文件夹下的多个文件
-
一个文件夹里面有很多个srt字幕文件,如何借助kimichat来自动批量合并呢?在kimichat对话框中输入提示词:你是一个Python编程专家,完成如下的编程任务:这个文件夹:D:\downloa...
- Java APT_java APT 生成代码
-
JavaAPT(AnnotationProcessingTool)是一种在Java编译阶段处理注解的工具。APT会在编译阶段扫描源代码中的注解,并根据这些注解生成代码、资源文件或其他输出,...
- Unit Runtime:一键运行 AI 生成的代码,或许将成为你的复制 + 粘贴神器
-
在我们构建了UnitMesh架构之后,以及对应的demo之后,便着手于实现UnitMesh架构。于是,我们就继续开始UnitRuntime,以用于直接运行AI生成的代码。PS:...
- 挣脱臃肿的枷锁:为什么说Vert.x是Java开发者手中的一柄利剑?
-
如果你是一名Java开发者,那么你的职业生涯几乎无法避开Spring。它如同一位德高望重的老国王,统治着企业级应用开发的大片疆土。SpringBoot的约定大于配置、SpringCloud的微服务...
- 五年后,谷歌还在全力以赴发展 Kotlin
-
作者|FredericLardinois译者|Sambodhi策划|Tina自2017年谷歌I/O全球开发者大会上,谷歌首次宣布将Kotlin(JetBrains开发的Ja...
- kotlin和java开发哪个好,优缺点对比
-
Kotlin和Java都是常见的编程语言,它们有各自的优缺点。Kotlin的优点:简洁:Kotlin程序相对于Java程序更简洁,可以减少代码量。安全:Kotlin在类型系统和空值安全...
- 移动端架构模式全景解析:从MVC到MVVM,如何选择最佳设计方案?
-
掌握不同架构模式的精髓,是构建可维护、可测试且高效移动应用的关键。在移动应用开发中,选择合适的软件架构模式对项目的可维护性、可测试性和团队协作效率至关重要。随着应用复杂度的增加,一个良好的架构能够帮助...
- 颜值非常高的XShell替代工具Termora,不一样的使用体验!
-
Termora是一款面向开发者和运维人员的跨平台SSH终端与文件管理工具,支持Windows、macOS及Linux系统,通过一体化界面简化远程服务器管理流程。其核心定位是解决多平台环境下远程连接、文...
- 预处理的底层原理和预处理编译运行异常的解决方案
-
若文章对您有帮助,欢迎关注程序员小迷。助您在编程路上越走越好![Mac-10.7.1LionIntel-based]Q:预处理到底干了什么事情?A:预处理,顾名思义,预先做的处理。源代码中...
- 为“架构”再建个模:如何用代码描述软件架构?
-
在架构治理平台ArchGuard中,为了实现对架构的治理,我们需要代码+模型描述所要处理的内容和数据。所以,在ArchGuard中,我们有了代码的模型、依赖的模型、变更的模型等,剩下的两个...
- 深度解析:Google Gemma 3n —— 移动优先的轻量多模态大模型
-
2025年6月,Google正式发布了Gemma3n,这是一款能够在2GB内存环境下运行的轻量级多模态大模型。它延续了Gemma家族的开源基因,同时在架构设计上大幅优化,目标是让...
- 比分网开发技术栈与功能详解_比分网有哪些
-
一、核心功能模块一个基本的比分网通常包含以下模块:首页/总览实时比分看板:滚动展示所有正在进行的比赛,包含比分、比赛时间、红黄牌等关键信息。热门赛事/焦点战:突出显示重要的、关注度高的比赛。赛事导航...
- 设计模式之-生成器_一键生成设计
-
一、【概念定义】——“分步构建复杂对象,隐藏创建细节”生成器模式(BuilderPattern):一种“分步构建型”创建型设计模式,它将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建...
- 构建第一个 Kotlin Android 应用_kotlin简介
-
第一步:安装AndroidStudio(推荐IDE)AndroidStudio是官方推荐的Android开发集成开发环境(IDE),内置对Kotlin的完整支持。1.下载And...
- 一周热门
-
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
飞牛OS入门安装遇到问题,如何解决?
-
系统C盘清理:微信PC端文件清理,扩大C盘可用空间步骤
-
10款高性能NAS丨双十一必看,轻松搞定虚拟机、Docker、软路由
-
python使用fitz模块提取pdf中的图片
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)