Linux:自动化构建 - make

Linux:自动化构建 - make

    • make基本概念
    • makefile语法
      • 变量
      • PHONY


make基本概念

make是一个用于自动化编译和构建过程的工具。它主要用于管理大型软件项目的构建过程,帮助开发者更高效地编译和部署代码,并减少人为错误的发生,这使得软件的编译和部署变得更加自动化和可靠。

其中make是一个命令,执行该命令需要当前操作目录下有一个名为makefile或者Makefile的文件,在makefile内部编写指令,随后就可以通过make快速执行一系列的指令了。

当前目录下有一个test.c文件,需要把该文件编译为可执行文件mybin。该过程就可以用make来自动执行。我先编写一个makefile,然后再讲解,代码如下:

mybin:test.c
    gcc -o mybin test.c
clean:
    rm -f mybin   

其中我们可以看到两条熟悉的指令:gcc -o mybin test.c,该指令完成把test.c文件编译为可执行文件mybinrm -f mybin,该指令完成对mybin的删除。

有了这个文件后,我们就可以直接输入make,完成对test.c的编译:

在这里插入图片描述

一开始目录下只有makefiletest.c两个文件,执行指令make后,Linux自动执行了指令gcc -o mybin test.c,最后目录中就出现了编译好的可执行文件mybin

我们也可以输入make clean,完成对mybin的删除:

在这里插入图片描述

执行make clean后,Linux自动执行了指令rm -f mybin,完成了mybin的删除。

看到其大致是如何执行后,我现在讲解以上代码中各个部分的作用是啥。

mybin:test.c
    gcc -o mybin test.c
  • mybin:test.c:这个由两个文件名通过一个:隔开的整体,叫做依赖关系。左侧的mybin叫做目标文件,右侧的test.c叫做依赖文件列表
  • gcc -o mybin test.c:这个语句叫做依赖方法

那么依赖关系目标文件依赖文件列表依赖方法之间有什么关系呢?

  1. 目标文件:

    • 目标文件是 make 命令要生成的文件,通常是可执行程序、库文件或中间目标文件。
    • 每个目标文件都有一条或多条命令来生成它。
    • 目标文件可以有多个依赖文件,这些依赖文件共同决定了目标文件的生成过程。
  2. 依赖文件列表:

    • 依赖文件列表描述了目标文件所依赖的输入文件,如源代码文件、头文件、库文件等。
    • 依赖文件列表告诉 make 哪些文件的变化会导致目标文件需要重新生成。
    • 依赖文件列表可以包含多个文件,用空格分隔。
  3. 依赖关系:

    • 目标文件与其依赖文件之间形成依赖关系。
    • 如果依赖文件中任何一个文件发生变化,目标文件就需要重新生成。
    • 依赖关系可以形成层次结构,实现文件之间的层次依赖。
  4. 依赖方法:

    • 依赖方法是生成目标文件所需执行的命令。
    • 依赖方法通常是 shell 命令,用于编译、链接等操作。
    • 依赖方法可以包含多条命令,每条命令用回车分隔。

这里要注意,依赖方法前面必须用一个Tab键隔开,不可以是四个空格。

再看到后半段:

clean:
    rm -f mybin   

该代码中,目标文件是clean,没有依赖文件列表,依赖关系是删除mybin的指令。只要执行make clean就会执行后面的代码rm -f mybin ,从而实现文件的删除。

左侧的目标文件clean叫做伪目标伪目标是 make 中的一种特殊目标,它不对应任何文件,只是用来执行一些命令。常见的伪目标有 all、clean、install 等。

那么现在有一个问题,为什么执行mybin:test.c直接使用make就可以,但是make clean才能执行clean呢?

make后面不接任何内容时,则在makefile中从上往下查找,找到第一个依赖关系执行

现有如下makefile

mybin:test.o
    gcc -o mybin test.o
test.o:test.s
    gcc -c -o test.o test.s
test.s:test.i
    gcc -S -o test.s test.i
test.i:test.c
    gcc -E -o test.i test.c

clean:
    rm -f test.i test.s test.o mybin

该文件完成了一个连续的C语言程序编译链接过程,但是其有一个问题在于,C语言编译过程应该依次生成.i.s.o文件,最后生成mybin。第一条依赖关系执行的时候,就缺少test.o文件,第二条依赖关系缺少test.s文件,以此类推,整个编译过程都是反的。那么该makefile可以执行成功吗?

我们试试:

在这里插入图片描述
可以看到,其执行成功了,这是为什么?我只输入了make指令,按理来说应该只执行第一条依赖关系,为什么除了clean的所有依赖关系都被执行了?

make会根据文件的依赖关系进行自动推导,以正确的顺序执行各个依赖关系

比如说一开始mybin缺少了test.o,于是makemakefile中查找,发现test.o:test.s的依赖关系可以生成test.o于是先执行该依赖关系。但是test.o:test.s的依赖关系缺少test.s,就再去找生成test.s的依赖关系,以此类推。


makefile语法

以上我们只简单讲解了一个基本的makefile例子,来帮助大家理解make,现在我们来讲解更多的makefile语法。

首先,在makefile中,指令内部的目标文件可以用$@代替,依赖文件列表可以用$^代替:

mybin:test.c
    gcc -o $@ $^  

在这里gcc -o $@ $^ gcc -o mybin test.c 是一样的。

变量

makefile中,是可以定义变量的,只需要用一个等号进行赋值即可:

bin=mybin
src=test.c

此时bin的值就是mybinsrc的值就是test.c,想要使用这个变量,就在对应的地方以$(变量名)的形式代替即可,这个变量可以放在任何地方,包括指令内部,依赖关系中等等。

比如下面这样:

bin=mybin
src=test.c

$(bin):$(src)
    gcc -o $(bin) $(src)

clean:
    rm -f $(bin)

任何需要test.cmybin的地方,都可以改为$(bin)$(src)


PHONY

讲解PHONY前,我们看到一个问题:

在这里插入图片描述
现在我们每次make都会生成一个mybin文件,如果直接gcc -o mybin test.c,那么新生成的mybin文件会覆盖原先的文件。但是make执行的指令,会进行一次检查,如果源文件test.c没有进行更新,那么make就会拒绝再次编译这个文件,因为编译出来的结果是一样的,导致mybin无法连续编译两次。

也就是上图中我们连续两次make,第二次拒绝了我们的原因。

PHONYMakefile中是一个非常有用的特殊目标。make执行到PHONY目标时,它会无条件执行该目标下定义的命令,而不会检查是否有同名的文件在

因此我们就可以对mybin使用这个PHONY,让它强制执行命令,不论当前目录下有没有mybin

修饰语法:

.PHONY:xxx

此时xxx对应的命令就会被强制执行。

makefile写法:

.PHONY:mybin
mybin:test.c
	gcc -o mybin test.c

此时mybin每次都会强制执行命令,不论之前是否存在mybin文件了。

但是这并不是PHONY的主要用途,其主要用于修饰伪目标。比如伪目标clean,如果当前目录下刚好有一个叫做clean的文件,那么clean作为一个目标文件,就有可能会被make禁止执行。可是我们的clean作为一个伪目标,其目的不是生成一个clean文件,而是执行命令,因此要用PHONY来强制执行clean对应的命令:

.PHONY: clean
clean:
    rm -rf *.o

在这里,clean是一个PHONY目标,即使当前目录下有一个名为clean的文件,make clean也会执行删除操作,而不是试图构建clean文件。


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

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

相关文章

Linux应用实战之网络服务器(五) 登录服务器初步调试

0、前言 准备做一个Linux网络服务器应用实战,通过网页和运行在Linux下的服务器程序通信,这是第五篇,编写服务器程序,与编写好的登录界面进行初步调试。 1、服务器编程 1.1 TCP服务器编程 在之前的登录界面中,我们指…

使用Docker部署开源项目FreeGPT35来免费调用ChatGPT3.5 API

Vercel部署FreeGPT35有严重限制,玩玩就好,真用还是得docker。 限制原因: Vercel的流式响应并不是一开始写流,客户端就能立刻收到响应流,而是先写到一个缓冲区,当流关闭才一股脑的流式响应回来(不是实时流) 因此导致: …

创意解决方案:如何将作品集视频集中于一个二维码或链接中?

引言:随着面试环节的进一步数字化,展示自己的作品集成为了求职过程中的重要一环。但除了使用传统的方式,如百度网盘或直接发送多个视频链接,有没有更便捷的方法将作品集的多个视频放在一个链接中呢? 本文将介绍一种创意解决方案…

系统监控-硬件资源-内存篇01-整体思路-性能指标-性能工具概览-Buffer/Cache

参考来源:性能优化实战 内存的功能主要用来存储系统和应用程序的指令、数据、缓存等。 内存性能分析整体思路 当你看到系统的剩余内存很低时,是不是就说明,进程一定不能申请分配新内存了呢?当然不是,因为进程可以使…

5.2 配置静态路由

5.2.1 实验1:配置IPv4静态路由 1、实验目的 通过本实验可以掌握: 配置带下一跳地址的IPv4静态路由的方法。配置带送出接口的IPv4静态路由的方法。配置总结IPv4静态路由的方法。配置浮动IPv4静态路由的方法。代理 ARP的作用。路由表的含义。扩展ping命…

数据仓库与数据挖掘(第三版)陈文伟思维导图1-5章作业

第一章 概述 8.基于数据仓库的决策支持系统与传统决策支持系统有哪些区别? 决策支持系统经历了4个阶段。 1.基本决策支持系统 是在运筹学单模型辅助决策的基础上发展起来的,以模型库系统为核心,以多模型和数据库的组合形成方案辅助决策。 它…

hal库实现串口通信——阻塞式 API

1STM32CobeMX设置 设置时钟源 rcc设置为外部时钟High Speed Clock (HSE)//设置为如图 再将其设置为72MHz 设置串口引脚为异步通信 设置波特率等 设置波特率范围提示点击波特率再点击图中的 我的设置 再打开中断 即可生成代码//省略项目设置 2代码设置 函数 HAL_UART_Trans…

探索数据中台的力量:企业数据资产管理的未来_光点科技

随着数字化时代的到来,"数据中台"这一概念逐渐为人所知,并迅速成为推动企业数据驱动转型的重要基石。数据中台不仅是集数据接入、管理、分析于一身的综合平台,更是企业实现数据资源集中管理、分析决策和业务创新的核心支撑。 一、数…

VRRP虚拟路由实验(华为)

思科设备参考:VRRP虚拟路由实验(思科) 一,技术简介 VRRP(Virtual Router Redundancy Protocol)是一种网络协议,用于实现路由器冗余,提高网络可靠性和容错能力。VRRP允许多台路由器…

如何提高旋转花键运行稳定性?

现代化精密仪器设备中,精密仪器的稳定工作性能对于生产效率和产品质量至关重要,运行效率和精度是常见问题。旋转花键作为机械传动系统中的重要组成部分,其稳定性也是直接影响到机械装配的质量和使用寿命,那么我们应该如何提升旋转…

uniapp开发小程序,通过缓存的方式,判断页面只弹出一次弹窗通知

一、需求 在使用uniapp开发小程序时,在【个人中心页面】-点击【我的推广】按钮进入详情页面时,要求出现【会员协议通知】的弹窗,并且有【确认和取消】两个按钮, 如果点了【取消】按钮,直接退出该页面,并且…

如何在极狐GitLab 使用Docker 仓库功能

本文作者:徐晓伟 GitLab 是一个全球知名的一体化 DevOps 平台,很多人都通过私有化部署 GitLab 来进行源代码托管。极狐GitLab 是 GitLab 在中国的发行版,专门为中国程序员服务。可以一键式部署极狐GitLab。 本文主要讲述了如何在[极狐GitLab…

Github Benefits 学生认证/学生包 新版申请指南

本教程适用于2024年之后的Github学生认证申请,因为现在的认证流程改变了很多,所以重新进行了总结这方面的指南。 目录 验证教育邮箱修改个人资料制作认证文件图片转换Base64提交验证 验证教育邮箱 进入Email settings,找到Add email address…

官网下载IDE插件并导入IDE

官网下载IDEA插件并导入IDEA 1. 下载插件2. 导入插件 1. 下载插件 地址:https://plugins.jetbrains.com/plugin/21068-codearts-snap/versions 说明:本次演示以IDEA软件为例 操作: 等待下载完成 2. 导入插件 点击File->setting->Pl…

HTML和markdown

总体情况 <p>在html的用处 在vscode中使用markdown [Markdown] 使用vscode开始Markdown写作之旅 - 知乎

STL库 —— list 的编写

目录 一、成员变量 ​编辑 二、push_back 函数 三、迭代器 iterator 3.1 iterator 结构体 3.2 begin() 与 end() 函数 3.3 iterator 运算符重载 3.4 -> 的重载 3.5 const_iterator 四、测试代码 五、修饰符成员 5.1 insert 函数 5.2 erase 函数 5.3 push 函数…

【Ubuntu】update-alternatives 命令详解

1、查看所有候选项 ​​​​​​​sudo update-alternatives --list java 2、​​​​​​​更换候选项 sudo update-alternatives --config java 3、自动选择优先级最高的作为默认项 sudo update-alternatives --auto java 4、删除候选项 sudo update-alternatives --rem…

Netty 入门应用之Http服务WebSocket

Netty实现Http服务 主要的变化是在初始化器中引入了新的编解码器 一些创建的类作用和Netty HelloWorld的小demo一样我这里就不再次重复了 1、Http服务端代码 public class HttpServer {public static void main(String[] args) {// 创建Reactor// 用来管理channel 监听事件 …

FME学习之旅---day22

我们付出一些成本&#xff0c;时间的或者其他&#xff0c;最终总能收获一些什么。 教程&#xff1a;栅格入门 FME 支持读取和写入 70 多种栅格格式。本教程将介绍几个基本示例&#xff0c;展示如何使用 FME 读取、转换和写入栅格数据。 FME 数据检查器不应用任何对比度增强。因…

实战项目——智慧社区(一)

1、项目介绍 系统功能 登录、修改密码、登出 &#xff08;1&#xff09;首页 &#xff08;1.1&#xff09;数据统计&#xff1a;小区人员统计对比图&#xff0c;占比图 &#xff08;2&#xff09;物业管理 &#xff08;2.1&#xff09;小区管理&#xff1a;小区数据的增删改…