一文入门CMake

我们前几篇文章已经入门了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啦,感觉不过瘾的小伙伴可以去官方文档进行更深一步的学习。英语不好的我快看吐了。。。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Spring底层原理之bean的加载方式四 @import 注解

bean的加载方式四 import 第四种bean的导入方式 是import导入的方式 在配置类上面加上注解就行 package com.bigdata1421.config;import com.bigdata1421.bean.Dog; import org.springframework.context.annotation.Import;Import(Dog.class) public class SpringConfig4 {…

Linux高级编程——进程

1.进程的含义? 进程是一个程序执行的过程,会去分配内存资源,cpu的调度 PID, 进程标识符 当前工作路径 chdir umask 0002 进程打开的文件列表 文件IO中有提到 (类似于标准输入 标准输出的编号,系统给0,1&#xf…

点云处理实战 点云平面拟合

目录 一、什么是平拟合 二、拟合步骤 三、数学原理 1、平面拟合 2、PCA过程 四、代码 一、什么是平拟合 平面拟合是指在三维空间中找到一个平面,使其尽可能接近给定的点云。最小二乘法是一种常用的拟合方法,通过最小化误差平方和来找到最优的拟合平面。 二、拟合步骤…

1.1章节print输出函数语法八种 使用和示例

1.打印变量和字符串 2-4.三种使用字符串格式化 5.输出ASCLL码的值和中文字符 6.打印到文件或其他对象(而不是控制台) 7.自定义分隔符、和换行符和结束符 8.连接符加号连接字符串 在Python中,print() 函数用于在控制台上输出信息。这是一个非常…

设置日历程序

目录 一 设计原型 二 后台源码 一 设计原型 二 后台源码 namespace 设置日历 {public partial class Form1 : Form{public Form1(){InitializeComponent();}private void dateTimePicker1_ValueChanged(object sender, EventArgs e){richTextBox1.Text dateTimePicker1.T…

React@16.x(42)路由v5.x(7)常见应用场景(4)- 路由切换动画

目录 1,实现路由切换基础样式 2,使用 CSSTransition 添加动画1,自定义动画组件 *TransitionRoute.jsx*2,*App.jsx*3,样式改动 3,注意点 通过一个例子来说明如何实现。 1,实现路由切换 基础样式…

prompt:我是晚餐盲盒,只要你问出“今晚吃什么”我就将为你生成美妙的食物推荐。

使用方法:在ChatGP粘贴下面提示词模型,点击输出。然后再问“晚餐有什么好吃的?”,AI输出丰种食物供你选择。抽到什么吃什么,极大的解决选择困难的问题。 客户需要生成1000条俏皮灵动,趣味盎然,比…

搭建 MySQL MHA

搭建 MySQL MHA 搭建 MySQL MHA实验拓扑图实验环境实验思路MHA架构故障模拟 实验部署数据库安装主从复制部署时间同步主服务器配置从服务器配置创建链接 MHA搭建安装依赖的环境安装 node 组件安装 manager 组件配置无密码认证在 manager 节点上配置 MHA管理 mysql 节点服务器创…

UI(二)控件

文章目录 PatternLockProgressQRCodeRadioRatingRichTextScollBarSearchSelectSlideSpanStepper和StepperItemTextTextAreaTextClockTextInputTextPickerTextTimerTimePickerToggleWeb PatternLock PatternLock是图案密码锁组件,以九宫格图案的方式输入密码&#x…

RTDETR更换优化器——Lion

RTDETR更换Lion优化器 论文:https://arxiv.org/abs/2302.06675 代码:https://github.com/google/automl/blob/master/lion/lion_pytorch.py 简介: Lion优化器是一种基于梯度的优化算法,旨在提高梯度下降法在深度学习中的优化效果…

vue3中通过vditor插件实现自定义上传图片、录入echarts、脑图、markdown语法的编辑器

1、下载Vditor插件 npm i vditor 我的vditor版本是3.10.2,大家可以自行选择下载最新版本 官网:Vditor 一款浏览器端的 Markdown 编辑器,支持所见即所得(富文本)、即时渲染(类似 Typora)和分屏 …

软件必须要进行跨浏览器测试吗?包括哪些内容和注意事项?

随着互联网的普及和发展,用户对软件的要求越来越高。无论是在台式机、笔记本还是移动设备上,用户都希望能够以最好的体验来使用软件。然而,不同的浏览器在解析网页的方式、支持的技术标准等方面存在差异,这就导致了同一个网页在不…

使用J-Link Commander查找STM32死机问题

接口:PA13,PA14,请勿连接复位引脚。 输入usb命令这里我已经连接过了STM32F407VET6了。 再输入connect命令这里我已经默认选择了SWD接口,4000K速率。 可以输入speed 4000命令选择4000K速率: 写一段崩溃代码进行测试: void CashCode(void){*((volatil…

(2024,RNN,梯度消失和爆炸,记忆诅咒,重参数化和动态学习率,权重矩阵对角化,复值 RNN)梯度消失和爆炸并不是故事的结局

Recurrent neural networks: vanishing and exploding gradients are not the end of the story 公和众与号:EDPJ(进 Q 交流群:922230617 或加 VX:CV_EDPJ 进 V 交流群) 目录 0. 摘要 1. 梯度消失和梯度爆炸 2. 记…

PyCharm2024 for mac Python编辑开发

Mac分享吧 文章目录 效果一、下载软件二、开始安装1、双击运行软件(适合自己的M芯片版或Intel芯片版),将其从左侧拖入右侧文件夹中,等待安装完毕2、应用程序显示软件图标,表示安装成功3、打开访达,点击【文…

深入解读一下`android.os.CountDownTimer`

简介 在 Android 开发中,CountDownTimer 是一个非常有用的类,它可以用于倒计时任务,比如倒计时器、限时活动等。CountDownTimer 提供了一个简单的方式来实现定时操作,无需我们手动管理线程和计时器。 本文将深入解析 CountDownT…

大数据------JavaWeb------Maven(完整知识点汇总)

额外知识点 IDE IDE是集成开发环境的缩写,它是一种软件应用程序,提供了编码、调试和部署软件的一站式解决方案。这些功能集成在一起,使开发人员能够在一个环境中完成整个软件开发过程,从编写代码到调试和测试,直到最终…

中霖教育:税务师考试通过率怎么样?

税务师考试的通过率通常在20%至30%的范围内,涵盖五个科目:《税法一》、《税法二》、《财务与会计》、《涉税服务实务》和《涉税服务相关法律》,成绩有效期为五年。 针对税务师备考,有效的学习策略至关重要。 1、熟悉各科的题型和…

什么是 SSH(安全外壳协议)以及如何工作

安全外壳协议(Secure Shell,简称SSH),旨在取代未加密的协议(如 Telnet 和 RSH)和未受保护的文件传输协议(如 FTP 和 RCP),在两个设备之间提供安全的加密连接。 安全外壳…

数字社交的领航者:解析Facebook的引领作用

在当今数字化社会中,社交网络已经成为了人们日常生活不可或缺的一部分。而在众多社交平台中,Facebook凭借其巨大的用户基础和创新的技术应用,被公认为数字社交领域的领航者之一。本文将深入解析Facebook在数字社交中的引领作用,探…