当前位置: 首页 > news >正文

【Linux】自动化构建工具-make/Makefile

个人主页 : zxctscl
如有转载请先通知

文章目录

  • 1. 前言
  • 2. 认识make/Makefile
  • 3. 了解make/Makefile原理
    • 3.1 依赖关系和依赖方法
    • 3.2 make检测的顺序
    • 3.3 PHONY:XXX
  • 4. makefile内置符号

1. 前言

在上一篇中已经了解了【Linux】编译器-gcc/g++使用,这次来一起看看make/Makefile。

2. 认识make/Makefile

make是一个命令
makefile是一个文件。

先touch一个makefile/Makefile(m大小写都可以)文件:
在这里插入图片描述
然后进入makefile写一段代码:
未来形成的可执行程序是mytest,依赖的是test.c。就是将test.c编译形成mytest的可执行程序。
在这里插入图片描述
直接make编译一下:
在这里插入图片描述
因为版本比较低,这里提示要加上-std=c99。重新打开makefile加上就行:
在这里插入图片描述

此时在重新make一下就有了:
会自动形成我们要的可执行程序mytest
在这里插入图片描述
在这里插入图片描述
如果想要清理编译产生的临时文件怎么办?
再次打开makefile:
在这里插入图片描述

退出后直接用命令清理一下

make clean

发现mytest已经没有了:
在这里插入图片描述

  1. makefile文件中,保存了编译器和链接器的参数选项,并且描述了所有源文件之间的关系。

    1. make程序会读取makefile文件中的数据,然后根据规则调用编译器,汇编器,链接器产生最后的输出。
  2. Makefile里主要包含了五个东西:显式规则、隐晦规则、变量定义、文件指示和注释。
    显式规则说明了,如何生成一个或多个目标文件。

  3. make有自动推导的功能,所以隐晦的规则可以让我们比较粗糙地简略地书写makefile,比如源文件与目标文件之间的时间关系判断之类

  4. 在makefile中可以定义变量,当makefile被执行时,其中的变量都会被扩展到相应的引用位置上,通常使用 $(var) 表示引用变量文件指示。包含在一个makefile中引用另一个makefile,类似C语言中的include;

  5. 注释,makefile中可以使用 # 在行首表示行注释

  6. 默认的情况下,make命令会在当前目录下按顺序找寻文件名为“GNUmakefile”、“makefile”、“Makefile”的文件,

3. 了解make/Makefile原理

打开Makefile,来看看前面两行:
在这里插入图片描述
写好Makefile后,当我们实际是在运行make时候:对应的程序就会在当前程序找makefile,然后读取makefile里面的内容,根据依赖关系,知道test.c形成mytest这样的可执行程序,根据gcc写的依赖方法来执行。

make会根据makefile的内容,完成编译或者清理工作。

3.1 依赖关系和依赖方法

怎么理解依赖关系和依赖方法呢?

举个例子:在现实生活中求人办事,就先得和他们产生某种关系,然后才会有后面的方法。就像小明没有生活费了,向他父亲打电话,就说一句:我是你儿子。这个就表明了依赖关系。表明了依赖关系然后,小明想要做什么?所以小明重新打电话说:我是你儿子,没钱了,打点钱。就有了依赖方法。
计算机要形成某种可执行程序,就得先表明依赖关系,然后怎么具体形成可执行程序,这就要有具体依赖方法。

依赖关系和依赖方法是完成一件事情的必然要素,并非makefile特有的。

在这里插入图片描述
这里clean:是一种特殊的依赖关系,这个clean不依赖任何一个文件,是一种特殊情况。
所以说依赖文件列表可以为空。
在这里插入图片描述

3.2 make检测的顺序

在形成可执行程序的时候直接make就可以直接形成mytest,而不是执行clean?
要执行clean,就必须这样写make clean
在这里插入图片描述

也可以用make mytest这样去运行:
在这里插入图片描述
makefile它的运行推导规则是:默认从上到下,对makefile文件进行扫描,默认形成第一个目标文件。也就是只有make的时候,默认形成第一个目标文件。

如果把clean放在前面,那么默认先执行的就是clean:
在这里插入图片描述
发现直接make的时候先执行的clean:
在这里插入图片描述

默认将执行的可执行程序放在前面,这样make的时候就能直接形成了。
所以make检测的顺序是从上往下的

3.3 PHONY:XXX

当我们在程序里面不加上:在这里插入图片描述
在这里插入图片描述

来make一下后,在make一下,发现就不行了:
在这里插入图片描述
在程序不被修改时,make后,默认就不会在形成新的可执行程序,它认为没有必要。
如果想要makefile里面的操作总是被执行,不要因为是最新的就拦截,就加上:.PHONY:mytest
在这里插入图片描述
make多少次都会执行,没有拦截。
在这里插入图片描述

.PHONY:XXX

XXX对应的方法,总是要被执行。

一般可执行程序不需要.PHONY:修饰,但是我们总是希望clean重新清理。
默认将clean用.PHONY:修饰:
要然清理工作总是被执行。
在这里插入图片描述
在这里插入图片描述
为什么makefile对最新的可执行程序,默认不重新形成呢?
如果在一个项目里面存在上千个源文件,每次改代码时候,可能就修改一小部分。做了改动之后,如果每一次都把所以的源文件重新编译一遍,就会带来效率的延缓。还有可能出现在没有修改情况下,一编译就重新执行,上千个文件,在编译的时候又得重新编译,又得花费很长时间。既然形成的可执行程序都是新的,那么就不需要再重新编译。
就是为了提高编译效率

那么是怎么做到不重新执行的呢?换句话说makefile怎么知道我的程序需要被编译呢?

这里有一个Modify:文件的最新修改时间:
在这里插入图片描述
源代码和可执行程序最近一次形成或者修改的时间一定是不一样的。
每一次都是先写源代码再形成可执行程序。
最终要判断程序是否被编译:只要对比,可执行文件最近修改的时间和源文件最近修改的时间,谁更新。
可执行程序最新就不需要再编了,源文件最新就重新编译一下。

这里源文件没有更新,就不能再编译:
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

来修改一下test.c的Modify时间:

touch test.c

在这里插入图片描述
make一下就又可以编了:
在这里插入图片描述

4. makefile内置符号

在这里插入图片描述

$:相当于取内容。
^:代表整个依赖文件列表,就是这里的code.c
@:代表目标文件,就是这里的code.exe

在编译的时候makefile会自动进行符号替换,把@替换为目标文件,^替换为code.c:
在这里插入图片描述
在这里插入图片描述

make之后,执行一下code.exe:
在这里插入图片描述

在把它清理掉:
在这里插入图片描述
在这里插入图片描述
把makefile如果要把它写全,就是这样的:
在这里插入图片描述
实际上编译器可以直接一步到位。
上面有4组依赖关系,就有依赖方法:

  1 code.exe:code.o2   gcc code.o -o code.exe -std=c993 code.o:code.s4   gcc -c code.s -o code.o -std=c995 code.s:code.i6   gcc -S code.i -o code.s -std=c997 code.i:code.c8   gcc -E code.c -o code.i -std=c99

在这里插入图片描述

在make后,在makefile里面写的代码重新编译了,编译后,依此形成了临时文件,并且形成了最终的可执行程序:

在这里插入图片描述
运行一下:
在这里插入图片描述

重新进入makefile:

  1 code.exe:code.o2   gcc code.o -o code.exe -std=c993 code.o:code.s4   gcc -c code.s -o code.o -std=c995 code.s:code.i6   gcc -S code.i -o code.s -std=c997 code.i:code.c8   gcc -E code.c -o code.i -std=c999 .PHONY:clean10 clean:11   rm -f code.i code.s code.o code.exe

在这里插入图片描述
此时code.i code.o code.s code.exe都被清理了:
在这里插入图片描述
在这里插入图片描述
发现执行的顺序和makefile里面代码顺序是相反的。
在这里插入图片描述
make拿到makefile文件时候,从上往下扫描,先看到的文件是code.exe,先识别的依赖关系是code.exe:code.o,但是发现code.o并不存在,所以 gcc code.o -o code.exe -std=c99是不执行的。
然后继续找下一组依赖关系:code.o:code.s,发现code.s并不存在,所以不执行这个依赖方法。
一直这样找,直到code.i:code.c,这个code.c是存在的,那么就会执行这一组依赖关系的依赖方法: gcc -E code.c -o code.i -std=c99。那么就形成了code.icode.i形成了,再回到code.s:code.i执行他的依赖方法gcc -S code.i -o code.s -std=c99。一直像这样,直到形成code.exe

makefile/make 会自动根据文件中的依赖关系,进行自动推导,帮助我们执行所有相关的依赖方法。

这些依赖关系就像栈一样,出栈执行方法:
在这里插入图片描述

如果注释调最后一组依赖关系:
在这里插入图片描述
就会出现没有规则去制作code.icode.i不存在,那么code.s也就不存在。
在这里插入图片描述

如果乱序排这些依赖关系:
在这里插入图片描述
能正常执行:
在这里插入图片描述

那么如果让第一个依赖关系放在后面呢:
在这里插入图片描述
此时make就不能执行:
在这里插入图片描述
因为makefile必须把最重要的文件放在前面,总的有个开头。一定要保证最终实现的目标文件是第一个。

那么之后我们写makefile就要这样写吗?
其实没有必要这样写。直接这样写就行:

  1 bin=code.exe2 src=code.c34 $(bin):$(src)5   gcc -o $@ $^ -std=c996 .PHONY:clean7 clean:8   rm -f $(bin)

在这里插入图片描述
这样也是可以执行的:
在这里插入图片描述

在这里插入图片描述
这里定义的变量就相当于宏。要改形成的可执行程序,就只改前面这两行的文件名就行。
举个例子:
在这里插入图片描述
重新make一下:
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
如果不想在执行命令的时候出现提示出像下面这样的命令的内容:
在这里插入图片描述

在这里插入图片描述
不想显示出这些命令,那么就在这些命令前面加上@:
在这里插入图片描述
此时发现已经不显示了:
在这里插入图片描述

一般我们所使用不仅仅只有一个依赖方法,可以继续往后跟:
在这里插入图片描述

在这里插入图片描述
有问题请指出,大家一起进步!!!

相关文章:

【Linux】自动化构建工具-make/Makefile

个人主页 : zxctscl 如有转载请先通知 文章目录 1. 前言2. 认识make/Makefile3. 了解make/Makefile原理3.1 依赖关系和依赖方法3.2 make检测的顺序3.3 PHONY:XXX 4. makefile内置符号 1. 前言 在上一篇中已经了解了【Linux】编译器-gcc/g使用,这次来一起…...

week07day03(power bi dax公式 零售数据业务分析)

一. 切片器(筛选)相关的三个函数 1.all (all后面的数据意思是 不受其影响) #ALL 筛选的是 筛选器 或 切片器#计算 销售金额 ,并且 不受到 门店ID 控制 计算金额 CALCULATE(SUM(销售表[金额]),ALL(销售表[门店ID]))#计算 销售金额 &#x…...

rembg报错onnxruntime_providers_tensorrt.dll

报错: 2024-03-16 04:16:59.4413827 [E:onnxruntime:Default, provider_bridge_ort.cc:1534 onnxruntime::TryGetProviderInfo_TensorRT] D:\a_work\1\s\onnxruntime\core\session\provider_bridge_ort.cc:1209 onnxruntime::ProviderLibrary::Get [ONNXRuntimeErro…...

精酿啤酒:一口啤酒,一份享受

在繁华的都市生活中,我们总是匆匆忙忙,追求着各种目标和成就。然而,在这个过程中,我们往往忽略了生活的本质,那就是享受。而Fendi Club 啤酒,正是为那些追求品质生活的都市精英们量身打造的。 Fendi Club啤…...

git报: “fatal: detected dubious ownership in repository“

“fatal: detected dubious ownership in repository”的中文翻译是:“致命错误:检测到仓库中存在可疑的所有权问题”。 这句话意味着 Git 在检查代码仓库时发现所有权存在问题,可能是由于文件或目录的所有权与 Git 仓库预期的所有权不匹配。…...

代码随想录算法训练营第27天|93.复原IP地址、78.子集、90.子集二

目录 一、力扣93.复原IP地址1.1 题目1.2 思路1.3 代码1.4 总结 二、力扣78.子集2.1 题目2.2 思路2.3 代码2.4 总结 三、力扣90.子集二3.1 题目3.2 思路3.3 代码3.4 总结 一、力扣93.复原IP地址 (比较困难,做起来很吃力) 1.1 题目 1.2 思路 …...

Java微服务轻松部署服务器

我们在日常开发微服务之后需要再服务器上面部署,那么如何进行部署呢,先把微服务的各个服务和中间件以及对应的端口列举出来,都打包成镜像,以及前端代码部署的nginx,使用docker-compose启动,访问服务器nginx…...

Wordpress站点通过修改.htaccess 设置重定向实现强制 https 访问

要在WordPress站点上通过修改.htaccess文件实现强制HTTPS访问,您可以按照以下步骤进行操作: 登录到WordPress站点管理后台。 在文件管理器或通过FTP访问网站根目录,找到并打开名为 .htaccess 的文件。 在打开的文件中添加以下代码&#xf…...

人大金仓助力国家电网调度中心培养国产数据库专家人才

近日,为进一步提升调度自动化安全可靠水平,提高电网数据应用效能,人大金仓助力国家电网调度中心培养国产数据库专家人才。 调度自动化系统拥有海量电网数据资源,是支撑电网安全经济优质运行的重要保障。数据库是调度自动化系统稳定运行的基石之一,其结构严谨、运行稳定、扩展灵…...

什么是增强型SSL证书?购买一张需要多少钱?

增强型SSL证书是一种提供更高级别安全验证与用户信任度的网络安全工具,也被称为EV证书。相较于DV(域名验证)和OV(组织验证)证书,它通过严格的身份核实流程确保网站所有者的合法性和真实性。 首先&#xff0…...

C++:函数传参到函数执行结束发生了什么

首先要明确两个概念 函数实参的入栈从右向左栈区从高地址向低地址偏移 接下来看下面一段代码 void fun(int a,int b,int c){std::cout<<&a<<" "<<&b<<" "<<&c<<std::endl; } int main(){fun(1,2,3); }…...

QT中dumpcpp以及dumpdoc使用

qt中调用COM的方式方法有四种&#xff0c;参考解释在 Qt 中使用 ActiveX 控件和 COM (runebook.dev) 介绍dumpcpp的使用方法Qt - dumpcpp 工具 (ActiveQt) (runebook.dev)&#xff1a; 在安装好了的qt电脑上&#xff0c;通过powershell窗口来实现&#xff0c;powershell比cmd要…...

RPM与DNF的操作实践

这几课有三个目标&#xff1a; 第一步&#xff1a;先配置软件源 跳转到yum.repos.d目录&#xff0c;用vim创建一个openeuler_x84_64.repo文件。这个文件就是我们将会用到的软件源。 我们在里面添加这些东西&#xff0c;保存并退出即可。 然后&#xff0c;我们用yum list all就…...

车道线检测之LaneNet

论文&#xff1a;Towards End-to-End Lane Detection: an Instance Segmentation Approach Github&#xff1a;https://github.com/MaybeShewill-CV/lanenet-lane-detection?tabreadme-ov-file 论文提出一种车道线检测网络LaneNet&#xff0c;该网络以enet为主干网络结构&…...

MySQL连接数不足导致服务异常GetConnectionTimeoutException

文章目录 场景复现解决方案一、调整连接数二、优化程序 场景复现 已经上线正常运行的项目突然很多功能无法使用&#xff0c;查看程序日志发现MySQL报错&#xff0c;异常信息: Could not open JDBC Connection for transaction; nested exception is com.alibaba.druid.pool.Ge…...

软考76-上午题-【面向对象技术3-设计模式】-创建型设计模式01

一、创建型设计模式一览 二、创建型设计模式 2-1、创建型设计模式的概念 一个类创建型模式使用继承改变被实例化的类&#xff1b; 一个对象创建型模式将实例化委托给另一个对象。 对应java的new一个对象。 2-2、简单工厂模式&#xff08;静态工厂方法&#xff09; 简单工厂…...

Matlab 双目相机标定(内置函数)

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 相机标定的目的就是要找到从世界坐标转换为图像坐标所用到的投影P矩阵各个系数(即相机的内参与外参)。具体过程如下所述: 1、首先我们需要获取一个已知图形的图像(这里我们使用MATLAB所提供的数据)。 2、找到同…...

【博客7.4】缤果Qt5_TWS串口调试助手V2.0 (高级篇)

超级好用的Qt5_TWS耳机串口调试助手 开发工具: qt-opensource-windows-x86-5.14.2 (编程语言C) 目录 前言 一、软件概要&#xff1a; 二、软件界面&#xff1a; 1.App演示 三、获取 >> 源码以及Git记录&#xff1a; 总结 前言 串口调试助手支持常用的50bps - 10M…...

CSS案例-4.padding导航栏练习

效果 相关数据: 上边框:3px,颜色#ff8500 下边框:1px,颜色#edeef0 背景颜色:#fcfcfc 高度:41px 内边距20px 字体颜色#4c4c4c 知识点 盒子边框border 属性 作用 border-width 定义边框粗细,单位px border-style 边框的样式 border-color 边框颜色 边框样式...

5.1.4.2、【AI技术新纪元:Spring AI解码】Llama2 Chat

Llama2 Chat Meta 的 Llama 2 Chat 是 Llama 2 系列大型语言模型的一部分。它在基于对话的应用程序中表现出色,参数规模范围从 70 亿到 700 亿不等。利用公共数据集和超过 100 万次人类注释,Llama Chat 提供了上下文感知的对话。 通过从公共数据源获取的 2 万亿标记进行训练…...

RestClient

什么是RestClient RestClient 是 Elasticsearch 官方提供的 Java 低级 REST 客户端&#xff0c;它允许HTTP与Elasticsearch 集群通信&#xff0c;而无需处理 JSON 序列化/反序列化等底层细节。它是 Elasticsearch Java API 客户端的基础。 RestClient 主要特点 轻量级&#xff…...

【Python】 -- 趣味代码 - 小恐龙游戏

文章目录 文章目录 00 小恐龙游戏程序设计框架代码结构和功能游戏流程总结01 小恐龙游戏程序设计02 百度网盘地址00 小恐龙游戏程序设计框架 这段代码是一个基于 Pygame 的简易跑酷游戏的完整实现,玩家控制一个角色(龙)躲避障碍物(仙人掌和乌鸦)。以下是代码的详细介绍:…...

Java 8 Stream API 入门到实践详解

一、告别 for 循环&#xff01; 传统痛点&#xff1a; Java 8 之前&#xff0c;集合操作离不开冗长的 for 循环和匿名类。例如&#xff0c;过滤列表中的偶数&#xff1a; List<Integer> list Arrays.asList(1, 2, 3, 4, 5); List<Integer> evens new ArrayList…...

深入浅出:JavaScript 中的 `window.crypto.getRandomValues()` 方法

深入浅出&#xff1a;JavaScript 中的 window.crypto.getRandomValues() 方法 在现代 Web 开发中&#xff0c;随机数的生成看似简单&#xff0c;却隐藏着许多玄机。无论是生成密码、加密密钥&#xff0c;还是创建安全令牌&#xff0c;随机数的质量直接关系到系统的安全性。Jav…...

【JVM】- 内存结构

引言 JVM&#xff1a;Java Virtual Machine 定义&#xff1a;Java虚拟机&#xff0c;Java二进制字节码的运行环境好处&#xff1a; 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收的功能数组下标越界检查&#xff08;会抛异常&#xff0c;不会覆盖到其他代码…...

【解密LSTM、GRU如何解决传统RNN梯度消失问题】

解密LSTM与GRU&#xff1a;如何让RNN变得更聪明&#xff1f; 在深度学习的世界里&#xff0c;循环神经网络&#xff08;RNN&#xff09;以其卓越的序列数据处理能力广泛应用于自然语言处理、时间序列预测等领域。然而&#xff0c;传统RNN存在的一个严重问题——梯度消失&#…...

大语言模型如何处理长文本?常用文本分割技术详解

为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...

大数据学习(132)-HIve数据分析

​​​​&#x1f34b;&#x1f34b;大数据学习&#x1f34b;&#x1f34b; &#x1f525;系列专栏&#xff1a; &#x1f451;哲学语录: 用力所能及&#xff0c;改变世界。 &#x1f496;如果觉得博主的文章还不错的话&#xff0c;请点赞&#x1f44d;收藏⭐️留言&#x1f4…...

Unity | AmplifyShaderEditor插件基础(第七集:平面波动shader)

目录 一、&#x1f44b;&#x1f3fb;前言 二、&#x1f608;sinx波动的基本原理 三、&#x1f608;波动起来 1.sinx节点介绍 2.vertexPosition 3.集成Vector3 a.节点Append b.连起来 4.波动起来 a.波动的原理 b.时间节点 c.sinx的处理 四、&#x1f30a;波动优化…...

【笔记】WSL 中 Rust 安装与测试完整记录

#工作记录 WSL 中 Rust 安装与测试完整记录 1. 运行环境 系统&#xff1a;Ubuntu 24.04 LTS (WSL2)架构&#xff1a;x86_64 (GNU/Linux)Rust 版本&#xff1a;rustc 1.87.0 (2025-05-09)Cargo 版本&#xff1a;cargo 1.87.0 (2025-05-06) 2. 安装 Rust 2.1 使用 Rust 官方安…...