【Linux我做主】make和makefile自动化构建

make和makefile自动化构建

  • make和makefile自动化构建
  • github地址
  • 前言
  • 背景介绍
    • 为什么需要make和makefile?
  • make和makefile解析
    • 什么是make和makefile
    • 依赖关系和依赖方法
      • 核心语法结构
      • 简单演示
        • 编译
        • 清理
      • 多阶段编译示例
    • make时执行的顺序
      • 场景1:clean目标在前(特殊情况)
      • 场景2:总构建目标在前(一般情况)
    • make如何实现不编译未被修改的文件
      • 场景引入
      • make实现原理
        • stat命令查看文件的属性
        • 三种时间:
          • 1. Access Time (atime)
          • 2. Modify Time (mtime)
          • 3. Change Time (ctime)
          • **对比总结**
          • **实际应用场景**
          • 文件 = 文件属性 + 文件内容
        • touch命令会修改相关的时间
      • **操作时间戳的命令**
    • 项目清理
      • 编写清理规则
      • makefile中的特殊符号
    • 总结make的构建原理
  • 结语

make和makefile自动化构建

github地址

有梦想的电信狗

前言

在Linux环境下开发中,makemakefile是构建复杂项目的核心工具。本文将深入解析其工作原理,并通过实例演示如何编写高效的自动化构建规则。


背景介绍

为什么需要make和makefile?

回顾我们编译单个文件时,例如main.c,使用如下命令:

gcc main.c -o main

单个文件,我们如此编译尚且可以。

但是,当一个项目中有成百上千个源文件时,这样子手动编译难免有些不够优雅。

手动输入命令容易显得有些繁琐,且容易出错

makemakefile就是为了解决这样的问题的。

makemakefile的功能和优势:

  • 大型项目管理:当工程包含数百个源文件时,手动编译和链接效率极低。
  • 自动化编译:通过定义规则明确编译顺序和依赖关系,make命令可一键完成全量构建。
  • 跨平台协作makefileUnix/Linux世界的通用构建语言,被所有主流IDE支持。

make和makefile解析

什么是make和makefile

  • make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如:DelphimakeVisual C++nmakeLinuxGNUmake。可见,makefile都成为了一种在工程方面的自动化构建方法。

  • makefile是一个文件,是一个按照特定规则编写的文件,当makefile编写完成后,make可以识别文件中的内容进行相应的操作。

  • make是一条命令,是一个可执行程序。

  • makefile是一个当前目录下的文件,和make搭配使用,完成项目自动化构建。其中makefile命名为makefileMakefile均可,首字母M/m不区分大小写。

依赖关系和依赖方法

核心语法结构

target1: prerequisitesrecipe
target1: prerequisitesrecipe
target1: prerequisitesrecipe
# ......
targetn: prerequisitesrecipe
  1. 目标(Target):要构建的目标

    • 最终产物(可执行文件)或中间产物(.o文件)
    • 支持伪目标(.PHONY target)用于执行操作
    • 示例:.PHONY修饰clean, install等非文件目标
  2. 依赖项(Prerequisites):构建目标所需的依赖文件项

    • 构建目标所需的文件(如.c.o文件)
    • 显式声明:直接列出的依赖文件,支持使用通配符*
    • 隐式依赖:通过变量或通配符自动推导
    • 顺序敏感:依赖列表的顺序影响构建优先级
  3. 配方(Recipe):生成目标所用到的shell命令。

    • 必须以Tab开头(4空格会导致语法错误)
    • 每条命令在独立shell中执行,每条当前形成当前target所需要的方法,称为依赖方法。该方法一般是Linux下的命令。
    • 使用@前缀抑制命令回显

简单演示

编译

有以下源程序mycode.c
在这里插入图片描述
在这里插入图片描述
输入以下指令:我们的单个源程序就跑起来了。

make
./mycode

在这里插入图片描述

可以看到:

  • 我们不需要输入繁琐的gcc编译指令,只需要输入make就自动可以帮助我们编译完成。
  • 输入make的本质还是执行了gcc编译指令。
清理
  • makefile文件中定义了clean目标的依赖方法时,输入make clean即可完成程序文件的清理。
    在这里插入图片描述

多阶段编译示例

有如下makefile
在这里插入图片描述

  • 依赖链mycode.c → mycode.i → mycode.s → mycode.o → mycode
  • 编译流程:预处理→编译→汇编→链接
    在这里插入图片描述

make时执行的顺序

找到
未找到
make命令
查找Makefile
解析第一个规则
报错退出
构建依赖树
执行拓扑排序
遍历所有节点
需要构建?
执行配方
跳过
更新时间戳
$ make
  1. 从终极目标mycode开始逆向解析
  2. 检查mycode是否存在或需要更新
  3. 递归检查mycode.smycode.imycode.c的依赖关系
  4. 按依赖链顺序执行编译命令

场景1:clean目标在前(特殊情况)

在这里插入图片描述

场景2:总构建目标在前(一般情况)

在这里插入图片描述

观察以上结果,可以看到,只执行make时,却执行了clean
执行make mycode,又得到了正确的编译结果。这是为什么呢?

由上我们可以总结归纳出:

  • make后面要跟上构建的target名。
  • make后面什么都不跟时,默认寻找Makefile中第一个target的依赖关系,执行其依赖方法。
  • make targetNamemake后面跟上target的名字时,该命令可以无视makefile中的出现次序,构建指定为targeName的目标文件
  • 建议
    • 建议将要生成的可执行程序作为第一个依赖关系,这样只输入make即可完成构建
    • 建议将clean放在最后,防止他人的误操作。

make如何实现不编译未被修改的文件

场景引入

观察一下场景:

  • 我们在make一次之后,如果没有对源文件进行修改,就无法再次make了。
    在这里插入图片描述
    在这里插入图片描述
    这是因为没有必要
  • 如果源文件没有被修改,再次编译得到的结果还是一样的。
  • make设计出这样的功能,是为了提高编译效率节省编译的时间开销

那么,以上功能是如何实现的呢?

make实现原理

可以思考下,如果让我们自己实现这样的功能,我们的思路会是怎么样的?

思路大致如下:

  • 通常情况下,我们一定是由源文件编译形成可执行文件。先有源文件,才会有可执行文件

  • 因此,一般而言(不通过命令改变文件的修改时间),源文件的最近修改时间,一定比可执行文件的最近修改时间要更老!!!

  • 因此,只需要比较可执行程序的最近修改时间和源文件的最近修改时间即可。

    • (此处用.exe代替可执行程序,Linux下是编译形成的有可执行权限的文件)

    • 最近修改时间,.exe新于.c,说明源文件时老的,不需要重新编译

    • 最近修改时间,.exe老于.c,说明源文件时新的,需要重新编译

    • 而一般而言,.exe.c的最近修改时间是不会一样的,除非通过命令来统一修改。

  • 因此,如果我们更改了源文件,历史上曾经还有可执行文件,那么源文件的最近修改时间,一定比可执行程序要新!

而文件的最近修改时间属于文件属性的一部分。

stat命令查看文件的属性

在这里插入图片描述

**我们今天先只关注文件属性中的时间,其他的属性暂不展开叙述。**以下是简介:

Linux/Unix 文件系统中,每个文件都有三个与时间相关的重要属性,可以通过 stat 命令查看。以下是它们的详细解释:


三种时间:
1. Access Time (atime)

定义:文件最后一次被 访问(读取)的时间
触发场景
• 用 catless 查看文件内容
• 用 grep 搜索文件内容
• 程序读取文件(如脚本执行、软件加载配置文件)
例外:如果使用 mount -o noatime 挂载文件系统,访问时间不会更新(提高性能)


2. Modify Time (mtime)

定义文件内容最后一次被 修改 的时间
触发场景
• 用编辑器修改文件内容
• 重定向输出到文件(如 echo "text" > file
• 文件被程序写入新数据
注意:修改文件内容会同时更新 change time(见下文)

示例

echo "new content" > myfile.txt  # 修改时间和change time均更新

3. Change Time (ctime)

定义:文件 元数据(metadata)(属性)最后一次被修改的时间
触发场景
• 修改文件权限(chmod
• 修改文件所有者(chown
• 创建硬链接
• 重命名文件
修改文件内容(因为文件大小等元数据变化)
注意ctime 不可通过 touch 直接修改,只能通过元数据操作间接更新

示例

chmod 644 myfile.txt  # change time更新

对比总结
时间类型简称触发操作查看命令
Accessatime读取文件内容stat -c %x filename
Modifymtime修改文件内容stat -c %y filename
Changectime修改文件属性stat -c %z filename

实际应用场景
  1. 备份系统:通过 mtime 判断文件内容是否变化,决定是否需要备份。
  2. 日志分析:通过 atime 追踪可疑的文件访问记录。
  3. 调试问题:用 ctime 确认文件权限或所有权是否被意外修改。
  4. 恢复文件:结合三个时间戳分析文件历史操作。

文件 = 文件属性 + 文件内容

我们之前提到过,文件 = 文件属性 + 文件内容,由上可以总结为:

  • 修改文件内容后,Modify时间会改变
  • 修改文件属性后,Change时间会改变
  • 一项操作可能同时改变三种时间中的多个

三种时间,可以用acm来简化记忆。

touch命令会修改相关的时间

在这里插入图片描述


  • touch会修改文件属性,修改后make便可再次编译
  • make通过比较源文件和可执行文件最近修改时间的新旧来判定要不要进行编译
  • 比较时将属性中的时间转化为Linux时间戳进行比较

在这里插入图片描述


操作时间戳的命令

查看时间戳

stat filename	# 查看文件的属性,属性中包含有时间,filename代表系统中文件的名字

修改时间戳

touch -a filename  # 仅修改atime  即为Access时间
touch -m filename  # 仅修改mtime  即为Modify时间
touch -t 202301010000 filename  # 指定时间(格式:[[CC]YY]MMDDhhmm[.ss])

  • 高级知识

relatime:现代 Linux 默认使用 relatime(仅当 atimemtimectime 旧时才更新,平衡性能与准确性)
ext4 的纳秒精度ext4 文件系统支持时间戳的纳秒级记录

  • 注意:touch命令可通过-a ``-m选项单独修改一个文件的Access时间和Modify,但无法单独修改文件的change时间
  • touch filename指令:无参数的 touch 命令会修改文件的三个时间。

项目清理

编写清理规则

  • 实际上makefile的清理命令make clean我们已经一直在使用了。
    在这里插入图片描述
.PHONY: clean
clean:rm -f mycode
  • clean是我们的targetrm -f mycode是目标的依赖方法,clean的依赖方法也可以是其他的命令。

前文我们提到,make通过比较源文件和可执行文件最近修改时间的新旧来判定要不要进行编译,但如果我就是想让一个target在每次make时都重新进行编译,该怎么办呢?这便是.PHONY的作用。

  • .PHONY声明伪目标,避免与同名文件冲突
  • 强制执行:无论是否存在clean文件,make clean都会执行
  • .PHONY修饰的目标文件,只要输入make,总是会被执行
    在这里插入图片描述
  • 实际上不建议用.PHONY修饰总的目标文件,简直用.PHONY修饰clean

makefile中的特殊符号

makefile中有很多特殊符号,本文只介绍三种特殊符号:@ $^ $@

  • @:@用于修饰依赖方法,使make之后,执行的命令不在终端中回显。
  • $^ $@ 常用在依赖方法中:
    • $@用来表示依赖关系冒号左边的内容。
    • $^用来表示依赖关系冒号右边边的内容。

在这里插入图片描述


总结make的构建原理

  1. 查找Makefile:优先读取Makefilemakefile
  2. 确定终极目标:解析第一个目标(示例中的hello
  3. 依赖树分析:递归检查所有依赖项是否存在
  4. 时间戳比对:若依赖文件比目标文件新,则触发重新编译
  5. 增量构建:仅重新编译修改过的文件及其依赖链

结语

通过合理设计makefile,开发者可以:

  1. 实现一键式编译,提升开发效率
  2. 利用增量编译节省构建时间
  3. 规范项目的构建和清理流程

如果本文对你有帮助,欢迎点赞收藏,技术问题欢迎在评论区交流!
一键三连,好运连连!

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

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

相关文章

Qt 入门 5 之其他窗口部件

Qt 入门 5 之其他窗口部件 本文介绍的窗口部件直接或间接继承自 QWidget 类详细介绍其他部件的功能与使用方法 1. QFrame 类 QFrame类是带有边框的部件的基类。它的子类包括最常用的标签部件QLabel另外还有 QLCDNumber、QSplitter,QStackedWidget,QToolBox 和 QAbstractScrol…

JAVA学习-多线程

线程 线程(Thread)是一个程序内部的一条执行流程。 程序中如果只有一条执行流程,那这个程序就是单线程的程序。 线程的常用方法及构造器: Thread提供的常用方法public void run() 线程的任务方法public void start() 启动线程public String getName() …

Github 2025-04-19Rust开源项目日报 Top10

根据Github Trendings的统计,今日(2025-04-19统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Rust项目10Python项目1Rust: 构建可靠高效软件的开源项目 创建周期:5064 天开发语言:Rust协议类型:OtherStar数量:92978 个Fork数量:12000…

OpenLayers:视图变换的方法

一、什么是视图变换? 视图变换就是指视图的 extent(范围)、center(中心点)、zoom(缩放级别)、 resolution(分辨率)、rotation(旋转角)等参数发生…

数字孪生火星探测车,星际探索可视化

运用图扑构建数字孪生火星探测车,高精度还原外观与内部构造。实时映射探测车在火星表面的移动、探测作业及设备状态,助力科研人员远程监测、分析数据,为火星探索任务提供可视化决策支持。

【NLP 66、实践 ⑰ 基于Agent + Prompt Engineering文章阅读】

你用什么擦干我的眼泪 莎士比亚全集 工业纸巾 还是你同样泛红的眼睛 —— 4.19 一、⭐【核心函数】定义大模型调用函数 call_large_model prompt:用户传入的提示词(如 “请分析这篇作文的主题”),指导模型执行任务 client&…

黑马Java基础笔记-1

JVM,JDK和JRE JDK是java的开发环境 JVM虚拟机:Java程序运行的地方 核心类库:Java已经写好的东西,我们可以直接用。 System.out.print中的这些方法就是核心库中的所包含的 开发工具: javac(编译工具)、java&…

PR第一课

目录 1.新建 2.PR内部设置 3.导入素材 4.关于素材窗口 5.关于编辑窗口 6.序列的创建 7.视频、图片、音乐 7.1 带有透明通道的素材 8.导出作品 8.1 打开方法 8.2 导出时,需要修改的参数 1.新建 2.PR内部设置 随意点开 编辑->首选项 中的任意内容&a…

Xcode16 调整 Provisioning Profiles 目录导致证书查不到

cronet demo 使用的 ninja 打包,查找 Provisioning Profiles 路径是 ~/Library/MobileDevice/Provisioning Profiles,但 Xcode16 把该路径改为了 ~/Library/Developer/Xcode/UserData/Provisioning Profiles,导致在编译 cronet 的demo 时找不…

【更新完毕】2025华中杯C题数学建模网络挑战赛思路代码文章教学数学建模思路:就业状态分析与预测

完整内容请看文末最后的推广群 先展示文章和代码、再给出四个问题详细的模型 基于多模型下的就业状态研究 摘要 随着全球经济一体化和信息技术的迅猛发展,失业问题和就业市场的匹配性问题愈加突出。为了解决这一问题,本文提出了一种基于统计学习和机器学…

[HOT 100] 1964. 找出到每个位置为止最长的有效障碍赛跑路线

文章目录 1. 题目链接2. 题目描述3. 题目示例4. 解题思路5. 题解代码6. 复杂度分析 1. 题目链接 1964. 找出到每个位置为止最长的有效障碍赛跑路线 - 力扣(LeetCode) 2. 题目描述 你打算构建一些障碍赛跑路线。给你一个 下标从 0 开始 的整数数组 obst…

2025年KBS SCI1区TOP:增强天鹰算法EBAO,深度解析+性能实测

目录 1.摘要2.天鹰算法AO原理3.改进策略4.结果展示5.参考文献6.代码获取 1.摘要 本文提出了增强二进制天鹰算法(EBAO),针对无线传感器网络(WSNs)中的入侵检测系统(IDSs)。由于WSNs的特点是规模…

JavaScript数据类型简介

在JavaScript中,理解不同的数据类型是掌握这门语言的基础。数据类型决定了变量可以存储什么样的值以及这些值能够执行的操作。JavaScript支持多种数据类型,每种都有其特定的用途和特点。本文将详细介绍JavaScript中的主要数据类型,并提供一些…

性能比拼: Elixir vs Go(第二轮)

本内容是对知名性能评测博主 Anton Putra Elixir vs Go (Golang) Performance Benchmark (Round 2) 内容的翻译与整理, 有适当删减, 相关指标和结论以原作为准 这是第二轮关于 Elixir 和 Go 的对比测试。我收到了一份来自 Elixir 创作者的 Pull Request ,并且我认为…

接口自动化 ——fixture allure

一.参数化实现数据驱动 上一篇介绍了参数化,这篇 说说用参数化实现数据驱动。在有很多测试用例的时候,可以将测试用例都存储在文件里,进行读写调用。本篇主要介绍 csv 文件和 json 文件。 1.读取 csv 文件数据 首先创建 csv 文件&#xff…

`peft`(Parameter-Efficient Fine-Tuning:高效微调)是什么

peft(Parameter-Efficient Fine-Tuning:高效微调)是什么 peft库是Hugging Face推出的用于高效参数微调的库,它能在不调整模型全部参数的情况下,以较少的可训练参数对预训练模型进行微调,从而显著降低计算资源需求和微调成本。以下从核心功能、优势、常见微调方法、使用场…

编程常见错误归类

上一篇讲了调试&#xff0c;今天通过一个举例回忆一下上一篇内容吧&#xff01; 1. 回顾&#xff1a;调试举例 在VS2022、X86、Debug的环境下&#xff0c;编译器不做任何优化的话&#xff0c;下⾯代码执⾏的结果是啥&#xff1f; #include <stdio.h> int main() {int …

在windows上交叉编译opencv供RK3588使用

环境 NDK r27、RK3588 安卓板子、Android 12 步骤操作要点1. NDK 下载选择 r27 版本&#xff0c;解压到无空格路径&#xff08;如 C:/ndk&#xff09;2. 环境变量配置添加 ANDROID_NDK_ROOT 和工具链路径到系统 PATH3. CMake 参数调整指定 ANDROID_NATIVE_API_LEVEL31、ANDRO…

浅析MySQL事务锁

在 MySQL 中,事务锁是用于确保数据一致性和并发控制的重要机制。事务锁可以帮助防止多个事务同时修改同一数据,从而避免数据不一致和脏读、不可重复读、幻读等问题。 以下是 MySQL 事务锁的关键点总结: 事务锁:用于确保数据一致性和并发控制。锁的类型: 行级锁:InnoDB,粒…

vue3 文件下载(excel/rar/zip)

安装axios npm install axios 在项目中引入 import axios from axios; 1、get接口excel文件下载 const file_key ref() const downLoadExcel (value:any) > {//file_key.value value axios({method: "get",url: "/api/da/download_excel/",//url:…