一文入门CMake

news2024/12/28 4:43:49

我们前几篇文章已经入门了gcc和Makefile,现在可以来玩玩CMake了。

CMake和Makefile是差不多的,基本上是可以相互替换使用的。CMAke可以生成Makefile,所以本质上我们还是用的Makefile,只不过用了CMake就不用再写Makefile了,会更简单一些。

一般来说小项目我们直接写Makefile,大项目我们就用CMake。

CMake是跨平台的,写起来也比Makefile简单,所以我们还是很有必要学学CMake的。

可以参考官方文档

CMake Reference Documentation — CMake 3.16.9 Documentationicon-default.png?t=N7T8https://cmake.org/cmake/help/v3.16/

首先我们先来安装一下CMake。

sudo apt-get install cmake

再输入下面命令看看版本,能看到就是正常安装成功了。

cmake --version

接下来我们就可以开始写CMake了,写Makefile的时候我们直接建一个名字叫“Makefile”或者“makefile”然后开始写就行了,CMake也差不多,只不过我们的脚本文件需要叫“CMakeLists.txt”

首先我们先在CMakeLists.txt里开头写上一句。

cmake_minimum_required(VERSION 3.0)

这一句是什么意思呢?从名字我们也可以看个大概意思,就是说我们通过这一句可以指定我们CMake的最低版本(VERSION是固定的,上面3.0是指定的最低版本号)。

官方文档英文看不懂,机翻的稀里糊涂,但是我还是把截图放这边吧,以官方文档说的为准。

因为CMake是跨平台的,所以我们是可以把CMakeLists.txt连通项目文件一起发给别人,然后别人再调用CMake去执行生成对应系统的Makefile,再通过Makefile去编译生成可执行文件,问题就在这,每个人环境不一样,CMake的版本也不一样,如果我们在CMakeLists.txt里用到了高版本才有的语句,那么别人低版本的CMake是会报错的,因此我们可以指定CMake的最低版本来防止这种情况发生。

当然不加这句也可以,但是我们最好是加上。

除了上面的,我们再加一个。

project(test)

这个意思是给我们的项目起名为test。

除了起名之外还以添加别的设置,但是一般情况下我们设置个项目名就行了。

有了上面两行基础的设置之后(可有可无),接下来最关键的还是设置我们要生成的可执行文件以及对应的依赖。

add_executable(test test.c)

使用上面的命令,指定生成的可执行目标,再在后面加上对应的依赖,最简单的CMakeLists.txt就算是写好了。就像下面这样。

接着我们使用cmake,后面跟上执行的CMakeLists.txt文件的路径。

然后我们就可以看到CMake替我们生成了不少文件,但是最重要的是Makefile文件,所以我们还需要执行一下make。

可以看成make之后给我们正常编译生成出了可执行文件,也能正常执行。

test.c里就是普普通通的打印一句“Hello World”。

最简单的用法就如上面这样。

接下来我们深入一点点。

在Makefile中我们可以使用变量,在CMake中我们也是可以使用变量的,不过使用方式略有不同。

我们在CMake中定义变量,需要使用一个命令set。

我们现在来改一改test.c这个文件。

包含了一个头文件,并且调用了其中一个函数。那么我们要编译获取可执行文件的话就需要多一个依赖文件了。

接下来我们再使用变量来改改CMakeLists.txt

使用set来定义一个变量,变量名在set命令中第一个参数的位置,后续用黄框框出的是设置的变量的值,可以有多个,用分号;或者空格隔开。使用的时候和Makefile不一样,CMake中用的是大括号{ }。

接着我们cmake,make,执行,发现可以正常编译生成可执行文件。

set除了可以定义变量,还可以修改宏定义的值,在CMake中我们可以通过修改一些宏定义的值来做一些配置。后面我们碰到了再说。

在一般情况下,工具文件不会和主文件放在一起,那么我们把工具文件放到单独一个文件夹里。

我们在CMakeLists.txt中也需要做出对应的修改,我们需要指定出头文件的路径。

使用的命令是下面这个。

比如说我放在了my_tools这个文件夹里,那么我就像下面这样子写。

include_directories(${PROJECT_SOURCE_DIR}/my_tools)

其中 PROJECT_SOURCE_DIR 是一个宏定义,它表示的是我们执行cmake命令时提供的CMakeLists.txt所在的路径,我们提取出这个宏定义就可以拼接出我们文件所在的位置了(绝对路径)。

除了上面的命令,还有个命令可以给某个具体的文件指定头文件路径。

参数一指定的是分配头文件目录的文件。

参数二是模式,一共有三种,INTERFACE,PUBLIC,PRIVATE。

简单来说就是如果有别的文件包含了这个文件,那么在设置了PUBLC模式的情况下,包含了这个文件的别的文件也可以搜索这个命令指定的头文件目录,设置PRIVATE模式的话就不行。剩下INTERFACE这个模式,这个我不太理解,个人感觉和PRIVATE差不多,比PRIVATE更“私人一点”,没啥特殊需求咱用PUBLIC就行了。

这样我们就指定完头文件路径,CMake也可以正常识别到头文件了。但是这样还是有个问题,因为我把库文件的 .c 和 .h都挪到my_tools这个文件夹里了,而我的CMakeLists.txt是下面这样的。

头文件是能找到,但是头文件对应的源文件还是找不到,原因就是我们像上面这么写的话,CMake会在当前目录下找,而不是跑到头文件路径下找。

解决方案有三。

第一,头文件放一个文件夹里,源文件放一个文件夹里,这样路径也能对得上。

第二,修改变量中mytool1.c的路径。

第三,我们使用CMake中的命令来代替我们写死的源文件路径。

使用上面的file,用法很多,但是我们只看红框框出来的。file可以帮助我们到指定路径去搜索文件并且存到变量里。

我们一共需要给这个命令传入三个参数,第一个是GLOB或者GLOB_RECURSE,用来指定搜索的方式,GLOB只会在指定的路径下搜索(足够了),GLOB_RECURSE会在指定路径下递归寻找(费时间)。

第二个是存放结果的变量。

第三个是指定的路径以及寻找的文件类型。

光说比较空泛,我们直接修改一下我们的CMakeLists.txt。

参数一 GLOB表示让CMake只在我们指定的路径下寻找。

参数二 给一个参数来存放结果,看得出来和我们普通编程不太一样,这个变量可以不声明不初始化,直接给就行。

参数三 路径,可以通过通配符来指定我们要搜索的文件后缀。

这样改过之后,CMake就能正常运行了,生成的Makefile也能够生成可执行文件。

除了我们上面这样分开编写源文件和头文件,然后一起塞给主文件编译生成可执行文件,我们还可以先打包成库文件,再塞给主文件生成可执行文件。

我们先看看静态库怎么做。

使用上面这个命令,我们需要给三个参数。

第一个是指定我们制作的静态库的名字。

第二个我们直接给STATIC表示制作的是静态库。

第三就是把源文件塞进去。

现在我们一样是先改改CMakeLists.txt,把用不到的先注释了。

我们指定的静态库名称是tool,但是生成的是libtool.a ,也就是说CMake会自动帮我们生成静态库名字的前缀和后缀。

如果我们制作的是动态库,则只需要把第二个参数从STATIC改成SHARED即可,其他不用动。

动态库的后缀在Linux中是 .so,而静态库是 .a 。

我们注意到,库文件默认生成的位置是和CMakeLists.txt在同一级目录下的,本来CMake就会生成一堆中间文件,现在又整这一出,会显得我们的文件夹很乱,因此我们可以通过指定库文件输出路径来使得我们的目录变得清爽一点。

我们可以使用set来修改一个宏定义,进而对库文件的输出路径进行设置。这个宏定义就是LIBRARY_OUTPUT_PATH

set(LIBRARY_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/lib)

可以看到已经给我们输出到指定目录下了,甚至之前我都没这个文件夹,CMake还帮我建了个。

那我们已经制作出了静态库和动态库了,那么下一步就是要使用它们了。

先说怎么使用静态库。

说是使用,也可以说是链接,因为库文件的本质就是将多个工具文件编译成 .o之后再合到一起。因此我们要使用的话,第一就是包含对应的头文件(这样才知道有哪些函数可以使用),第二就是在链接阶段和主文件放一起生成可执行文件。

我们分三步走,第一步是指定头文件路径,第二步是指定库文件路径,第三步是指定链接的库文件。

指定头文件路径我们上面说过了,我们直接跳到第二步,用的是下面这个命令。

我们就像指定头文件路径那样把库文件路径塞进去就行。

下一步指定链接的库文件。

指定链接的库文件的时候我们指定的库文件名字可以是全称libxxx.a,也可以掐头去尾,就像我们制作库文件名字那样。

最终修改一下CMakeLists.txt。

接下来我们再链接一下动态库,链接动态库用的是另一个命令(指定库文件路径的命令和上面静态库的一样)。

第一个参数是我们要链接动态库的文件,一般也就是我们的可执行文件。

剩下的参数以两个为一组,每组添加一个动态库。两个参数依次是链接模式和链接的动态库名字。

模式有三种,INTERFACE,PUBLIC,PRIVATE。

我们请文心一言介绍一下吧。

在 CMake 中,target_link_libraries() 函数用于指定一个目标(如库或可执行文件)所依赖的其他库。当你链接库到另一个目标时,你经常需要指定这些依赖是如何被传播的。这就是 INTERFACEPUBLIC 和 PRIVATE 关键字的作用。它们用于指定库链接时的可见性和传播规则。

以下是这些关键字的详细解释:

  1. PRIVATE

    • 当一个库 A 使用 PRIVATE 关键字链接到另一个库 B 时,这意味着库 A 会直接链接到库 B,但库 A 的消费者(即链接到库 A 的其他目标)不会看到或链接到库 B
    • 简单来说,PRIVATE 链接是库 A 内部的事情,不会影响它的消费者。
  2. PUBLIC

    • 当一个库 A 使用 PUBLIC 关键字链接到另一个库 B 时,这意味着库 A 会直接链接到库 B,并且任何链接到库 A 的目标(如可执行文件或其他库)也会自动链接到库 B
    • 换句话说,库 A 的消费者会“继承”库 A 的 PUBLIC 依赖。
  3. INTERFACE

    • 当一个库 A 使用 INTERFACE 关键字链接到另一个库 B 时,这意味着库 A 的消费者(即链接到库 A 的其他目标)会链接到库 B,但库 A 本身不会直接链接到库 B
    • 这通常用于创建一个“接口”或“契约”,指定库 A 的消费者需要哪些额外的库,但库 A 本身在编译时并不需要这些库。

示例

假设你有三个库:LibALibB 和 LibC

  • LibA 使用 PUBLIC 链接到 LibB
  • 一个可执行文件 Exe 使用 LibA 和 LibC

在这种情况下:

  • LibA 会直接链接到 LibB
  • Exe 会自动链接到 LibA 和 LibB(因为 LibA 是 PUBLIC 链接到 LibB 的),但它不会自动链接到 LibC,除非你也指定了 Exe 需要链接到 LibC

使用这些关键字可以帮助你更精细地控制你的构建系统和依赖关系,特别是在创建库和框架时。

没什么特殊需求的话我们直接给个PUBLIC就行,甚至默认就是PUBLIC,我们不写都可以。

那么到此为止,我们就算入门CMake啦,感觉不过瘾的小伙伴可以去官方文档进行更深一步的学习。英语不好的我快看吐了。。。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1870293.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Zookeeper 四、Zookeeper应用场景

Zookeeper是一个典型的发布/订阅模式的分布式数据管理与协调框架,我们可以使用它来进行分布式数据的发布与订阅。另一方面,通过对Zookeeper中丰富的数据节点类型进行交叉使用,配合Watcher事件通知机制,可以非常方便地构建一系列分…

Day.js

Day.js 是什么? Day.js是一个极简的JavaScript库,可以为现代浏览器解析、验证、操作和显示日期和时间。 Day.js中文网 为什么要使用Day.js ? 因为Day.js文件只有2KB左右,下载、解析和执行的JavaScript更少,为代码留下更…

如何避免爬取网站时IP被封?

互联网协议 (IP) 地址是识别网络抓取工具的最常见方式。IP 是每个互联网交换的核心,对其进行跟踪和分析可以了解很多有关连接客户端的信息。 在网络抓取中,IP 跟踪和分析(又名指纹)通常用于限制和阻止网络抓取程序或其他不需要的访…

配电房挂轨巡检机器人

配电房作为电网中的重要组成部分。其运行的的安全和稳定性直接影响到电力供应的质量。然而,传统的人工巡检模式存在诸多弊端,例如巡检效率低下、人员安全难以保障、巡检结果主观性强等问题。为了解决这些问题,旗晟机器人推出B3系列升降云台轨…

浅谈LiveData的通知过程

浅谈 LiveData 的通知机制 LiveData 和 ViewModel 一起是 Google 官方的 MVVM 架构的一个组成部分。巧了,昨天分析了一个问题是 ViewModel 的生命周期导致的。今天又遇到了一个问题是 LiveData 通知导致的。而 ViewModel 的生命周期和 LiveData 的通知机制是它们的…

AI绘图软件:设计师的创意加速器

在人工智能的浪潮中,AI绘图软件工具已成为设计师和创意工作者的得力助手,它们不仅加速了复杂绘图任务的完成,还激发了无限创意。本文将为您介绍几款AI绘图软件工具,它如何成为提升工作效率和创意灵感的关键。 1. StartAI&#xf…

ISO37001认证:防范贿赂风险的强大工具

随着全球反贿赂法规的日益严格,各类组织面临的贿赂风险和合规压力不断增加。ISO37001反贿赂管理体系认证应运而生,为组织提供了一个系统化的管理框架,帮助其有效发现、预防和管控贿赂风险。本文将详细探讨ISO37001认证的作用和意义&#xff0…

小白学python(第三天)

小伙伴,大家好呀,昨天的内容吸收的好?昨天有小伙伴私信我,建议我在博文中加点练习题,可以看出这位童鞋很想学好这门语言哈,那我也尽量满足大家的要求。 从控制台输入 语法格式: 变量名 input…

神经网络学习笔记9-简单的反向传播和线性回归

tensor在pytorch中是一个非常重要的类型 假如需要计算梯度,就将tensor中 requires_grad设为true loss是一个张量,在做运算时构建运算图,因此不要直接进行,会将将梯度存入w,当反向传播完成 后,该计算图会…

C# 23设计模式备忘

创建型模式:单例(Singleton)模式:某个类只能生成一个实例,该类提供了一个全局访问点供外部获取该实例,其拓展是有限多例模式。 原型(Prototype)模式:将一个对象作为原型&…

地铁中的CAN通信--地铁高效安全运转原理

目前地铁采用了自动化的技术来实现控制,有ATC(列车自动控制)系统可以实现列车自动驾驶、自动跟踪、自动调度;SCADA(供电系统管理自动化)系统可以实现主变电所、牵引变电所、降压变电所设备系统的遥控、遥信、遥测;BAS(环境监控系统)和FAS(火灾报警系统)可以实现车站…

IDEA版本推荐

推荐版本: IDEA 2024.1.4 下载链接:IDEA下载 (下载时可以往下拖,选到自己想要的版本哦) 本人由于项目开发需要,陆续用过几个版本的IDEA,包括: IDEA 2020.2.4 。这是在看韩顺平老师…

六西格玛绿带可以跳过,直接学六西格玛黑带吗?真实情况告诉你

在现代企业管理中,六西格玛(Six Sigma)已经成为提升质量和效率的重要工具。对于很多企业而言,培养内部的六西格玛专家,特别是黑带(Black Belt),是推动持续改进的关键。然而&#xff…

十常侍乱政 | 第2集 | 愿领精兵五千,斩关入内,册立新君,诛杀宦党,扫清朝廷,以安天下 | 三国演义 | 逐鹿群雄

🙋大家好!我是毛毛张! 🌈个人首页: 神马都会亿点点的毛毛张 📌这篇博客是毛毛张分享三国演义文学剧本中的经典台词和语句,本篇分享的是《三国演义》第Ⅰ部分《群雄逐鹿》的第2️⃣集《十常侍乱政治》&am…

代码随想录算法训练营第三十六天|62.不同路径、 63. 不同路径 II、343.整数拆分(可跳过)、96.不同的二叉搜索树(可跳过)

62.不同路径 题目链接:62.不同路径 文档讲解:代码随想录 状态:还行 思路:当前状态的只有可能是从上面或者左边过来的,所以 dp[i][j] dp[i-1] dp[j-1] 题解: public int uniquePaths(int m, int n) {if (…

Android Kotlin 中的闭包函数

闭包函数是现代编程语言中一个重要的概念,Kotlin 作为一种现代的 JVM 语言,自然也支持闭包函数。本文将详细介绍闭包函数的概念、在Kotlin 中的使用方法,以及一些常见的应用场景。 什么是闭包函数? 闭包函数,也称为闭…

50etf期权合约一手多少钱你知道吗?

今天带你了解50etf期权合约一手多少钱你知道吗?50etf期权有不同价值的合约,每手50etf期权合约从几元到几百元再到上千元的都有,具体需要根据投资者选择了什么价值的合约。 50etf期权权利金 50ETF期权合约的权利金是买方需要缴纳的费用&…

上古世纪战争台服官网地址+台服预约+预创建角色教程

上古世纪战争台服上线啦,在《上古世纪战争》中,通过主要势力和地区,剧情和角色可以想起原作。《上古世纪战争》的主要背景为,原大陆消失之后,完成移民的种族们定居在诺伊大陆之后遇到的多个势力之间的冲突。同时&#…

视觉灵感的探索和分享平台

做设计没灵感?大脑一片空白?灵感是创作的源泉,也是作品的灵魂所在。工作中缺少灵感,这是每个设计师都会经历的苦恼,那当我们灵感匮乏的时候,该怎么办呢?别急,即时设计、SurfCG、Lapa…

ulimit报错

问题 执行命令“ulimit -c 2048 ”时报错:“bash: ulimit: core file size: cannot modify limit: Operation not permitted” 原因 权限不够。 解决办法 执行命令“sudo gedit /etc/security/limits.conf”打开文件limits.conf文件内添加内容如下:…