Linux 项目自动化构建工具:make/makefile

什么是 make

make 是一个命令,他会在源文件的当前目录下寻找 makefile 或者 Makefile 文件执行这个文件中的代码。

makefile 文件的编写

我们先来见见猪跑,看看 make 怎么用的:
下面是 makefile 文件的内容:
在这里插入图片描述
这是 test.c 中的内容:

#include<stdio.h>
int main()
{printf("hello make\n");return 0;
}

之前我们想要使用 gcc 编译 test.c 生成 test 可执行文件,你是不是要这样写命令:

gcc -o test test.c

在我们写了上面的 makefile 文件之后,我们就能使用 make 命令来代替啦!
在这里插入图片描述
我们可以看到使用 make 命令之后顺利编译出来了可执行文件,并且能够顺利执行!

makefile 文件的编写

我们再来看 makefile 文件中的代码:
在这里插入图片描述
其中这个冒号前面的部分叫做依赖关系(绿色框框的那个),后面的部分叫做依赖方法(红色框框的那个)。听上去十分高大尚,翻译成白话文就是:依赖关系的形成需要依赖方法中的所有文件。

make 命令会自动扫描 makefile 文件,查看当前目录下是否存在依赖方法中的所有文件,如果已经存在,那么就会执行下一行 Tab 缩进的代码(只能是 Tab 缩进)。那么如果不存在怎么办呢?

我们在讲 C 语言编译链接的时候知道:从 C 语言的源文件到生成功可执行文件是分成很多步骤的:

gcc -E
gcc -S
gcc -C

根据这个原理我们就在 makefile 文件中将那一行编译生成可执行文件的代码分成一步一步来执行。

test:test.ogcc test.o -o test
test.o:test.sgcc -c test.s -o test.o
test.s:test.igcc -S test.i -o test.s
test.i:test.cgcc -E test.c -o test.i

make 命令扫描 makefile 文件时:

  • 发现依赖关系 test 的依赖文件 test.o 在源文件的当前目录不存在,继续向下扫描。
  • 发现依赖关系 test.o 的依赖文件 test.s 在源文件的当前目录不存在,继续向下扫描。
  • 发现依赖关系 test.s 的依赖文件 test.i 在源文件的当前目录不存在,继续向下扫描。
  • 发现依赖关系 test.i 的依赖文件 test.c 在源文件的当前目录已经存在,就会执行 Tab 缩进的代码:gcc -E test.c -o test.i 生成 test.i
  • test.i 依赖文件已经存在啦,就会执行:gcc -S test.i -o test.s 生成 test.s 文件。
  • test.s 依赖文件已经存在啦,就会执行:gcc -c test.s -o test.o 生成 test.o 文件。
  • test.o 依赖文件已经存在啦,就会执行:gcc test.o -o test 生成 test 可执行文件。

在上述过程执行完成之后(使用 make 命令之后),源文件的当前目录下就会生成:test.i test.s test.o test 文件。
在这里插入图片描述
我们可以看到显示出来命令的执行顺序与我们推导的顺序是一样的哈!
综上所述:扫描 makefile 文件的时候,如果源文件的当前目录不存在依赖文件,就会递归似的向下执行,这种行为叫做 make 的自动化推导

清理可执行文件

我们在更改了源文件的代码之后,需要清除可执行文件后重新编译。那么清除可执行文件能否使用 make 命令呢?那肯定是可以的撒!

clean:rm -f test

其中,clean 是依赖关系,冒号右侧为空说明表明没有依赖的文件。那么我们应该如何使用这个依赖关系呢?
执行命令:make clean 即可。

make clean

在这里插入图片描述
我们看到顺利运行了呢!

clean 放在 makefile 文件的最开头

如果我们像这样写 makefile 文件会发生什么呢?

clean:rm -f test
test:test.cgcc -o test test.c

在这里插入图片描述
可以看到我们想要编译文件就需要使用命令:make test,而 make 命令变成了执行:rm -f test
由此可见:make 命令会从上到下扫描 makefile 文件,将扫描到的第一个依赖关系作为 make 命令的默认行为


不推荐将依赖关系 clean 放在 makefile 文件的开头


make 命令编译多个文件

多个源文件生成一个可执行程序

我们写一个代码:在 function.h 中声明一个 Add 函数,在 function.c 中实现 Add 函数,然后在 test.c 中调用 Add 函数。
function.h

#pragma once
int Add(int a, int b);

function.c

int Add(int a, int b)
{return a + b;
}

test.c

#include<stdio.h>
#include "function.h"
int main()
{int a, b;scanf("%d %d", &a, &b);printf("a + b 的结果:%d\n", Add(a, b));return 0;
}

我们想要编译 function.h function.c test.c 应该怎么做呢?其实很简单哈!

test:function.c test.cgcc -o test test.c function.c
clean:rm -f test

如果是多个源文件生成一个可执行程序,只需要在依赖文件中以空格隔开多个源文件即可。如果 .h 文件在源文件的当前目录,依赖文件中是不需要写 .h 文件的!

多个源文件生成多个可执行程序

如果在 makefile 文件的目录下有多个源文件,并且想要将这些个源文件分别编译成可执行文件应该怎么做呢?你可以先想一想🤔,你应该是有能力写出来的。
我们来写这样两个源文件:test1.ctest2.c
test1.c

#include<stdio.h>int main()
{printf("i am test1.c\n");return 0;
}

test2.c

#include<stdio.h>int main()
{printf("i am test2.c\n");return 0;
}

我们要使用 make 命令讲他们分别编译成:test1test2 两个可执行文件。makefile 文件可以这样写:

All:test1 test2
test1:test1.cgcc -o test1 test1.c
test2:test2.cgcc -o test2 test2.c
clean:rm -f test1 test2

依赖关系:All 依赖于 test1 和 test2make 命令扫描 makefile 文件,发现源文件当前目录不存在 test1 和 test2 那么就会继续向下扫描。当扫描到 test1 和 test2 这两个依赖关系,他们的依赖文件都在源文件的当前目录。可以直接执行他们 Tab 缩进的代码,生成 test1 和 test2,最后完成两个源文件的编译生成两个可执行文件。
在这里插入图片描述

我们可以看到执行 make 命令之后也是顺利生成了 test1test2 两个可执行文件了呢!

make 可以重复编译吗?为什么?

我们还是回到最开始的那个代码:
test.c

#include<stdio.h>
int main()
{printf("hello make!\n");return 0;
}

makefile

test:test.cgcc -o test test.c
clean:rm -f test

我们发现在不修改代码的情况下,是不允许二次编译的:
在这里插入图片描述
这是为什么呢?
显然是因为没有这个必要哈,既然你的源文件没有被修改为什么要为你重新编译呢?
那这个是怎么做到的呢?

  • 一般来说,我们都是先有源文件,再有可执行程序。这就意味着源文件的最近修改时间比可执行程序的最近修改时间要早。
  • 因此,我们只需要比较可执行程序的最近修改时间和源文件的最近修改时间,就可以判断源文件是否需要重新被编译啦!

🤔思考:源文件的最近修改时间会和可执行程序的最近修改时间想等吗?这个一般是不会的!😊


那么,这个用来比较的时间哪里来呢?
我们先来学习一个命令吧:这个命令可以查看一个文件的时间状态。

stat 文件

在这里插入图片描述
Access Modify Change 这三个时间称为文件的 ACM 时间。

  • Access:最近访问时间,几乎文件的任何操作之后,这个时间都会发生改变。
  • Modify:当对文件的内容做出修改之后,这个时间就会更新。
  • Change:当对文件的属性做出修改之后,这个时间就会更新。

这就意味着,一旦对文件的内容做出修改,Access Modify Change 时间都会被更新。

因为 Access 时间要被频繁被修改,在实际的实现中 Access 时间的更新是有一定的更新策略(例如:当 Modiify 或者 Change 时间到达一定的次数之后再更新 Access 时间),而不是根据 Access 时间的定义那样,操作一次文件都要更新这个时间。
原因:文件是被存放在磁盘中的,将数据刷新到磁盘的速度是比较慢的,频繁地修改 Access 时间势必会影响操作系统的效率的。

在判断源文件是否需要重新编译,就是根据源文件和可执行程序 Modify 时间的比较结果来判定的!


如何验证呢?
再来学习一个命令吧:

touch 文件名

这个 touch 命令除了能够创建一个普通文件,还有一个功能就是:当这个文件已经存在时,能更新一个文件的 ACM 时间到当前的最新时间。
因此,我们可以更新源文件的 ACM 时间到最新,使得 make 命令可以反复编译一个相同的源文件。
在这里插入图片描述我们看到,第一次可以顺利编译,这很正常。第二次使用 make 编译的时候就不能了!我们在更新源文件的 ACM 时间之后又能使用 make 编译了!由此可以验证就是通过比较源文件与可执行文件的时间来判断是否能使用 make 再次编译的!

如何让一个依赖关系一直被执行

我们上面讲了通过 touch 命令可以使用 make 一直编译。但是,还是不建议这么做,没有修改源文件就不要重复编译,这很好,不是吗!
但是清理可执行文件的依赖关系,我们就有这个需求,让他总是被执行。那么 makefile 文件应该怎么写呢?

test:test.cgcc -o test test.c
.PHONY:clean
clean:rm -f test

makefile 文件中被 .PHONY 修饰的依赖关系就可以被一直执行啦!
你若不信,就可以给可执行文件 test 这个依赖关系加上 .PHONY 修饰,看能不能 make 重复编译(不建议这么做!!!)。

特殊符号

  • $@:表示:依赖关系:依赖方法 中冒号前面的一坨!
  • @^:表示:依赖关系:依赖方法 中冒号后面的一坨!

那么,我们写 makefile 文件就可以这么写啦:

test:test.cgcc -o $@ $^
.PHONY:clean
clean:rm -f test

在这个 makefile 文件中:$@ 就是 test$^ 就是 test.c

==这才是我们在平时用的最多的 makefile 文件的编写方法啦!==😊

取消回显

我们在使用 make 的时候是不是能看到 make 推导出来的要执行的指令的内容!像这样:
在这里插入图片描述
如果你不想回显命令,只需要在指令前面加上 @ 符号就可以啦!

test:test.c@gcc -o $@ $^
.PHONY:clean
clean:@rm -f test

在这里插入图片描述

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

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

相关文章

WPF创建进度条

使用wpf做一个原生的进度条&#xff0c;进度条上面有值&#xff0c;先看效果。 功能就是点击按钮&#xff0c;后台处理数据&#xff0c;前台显示处理数据的变化&#xff0c;当然还可以对进度条进行美化和关闭的操作&#xff0c;等待后台处理完毕数据&#xff0c;然后自动关闭。…

Python入职某新员工大量使用Lambda表达式,却被老员工喷是屎山

Python中Lambda表达式是一种简洁而强大的特性,其在开发中的使用优缺点明显,需要根据具体场景权衡取舍。 Lambda表达式的优点之一是它的紧凑语法,适用于一些短小而简单的函数。这种形式使得代码更为精炼,特别在一些函数式编程场景中,Lambda表达式可以提高代码的表达力。此外…

DMX512协议及对接口电路的分析

1、DMX512协议简介 DMX 是Digital MultipleX 的缩写&#xff0c;意为多路数字传输(具有512条信息的数字多路复用”)。DMX512控制协议是美国舞台灯光协会(usITT)于1990年发布的灯光控制器与灯具设备进行数据传输的工业标准&#xff0c;全称是USITTDMX512(1990); DMX512 在其物理…

福州大学《嵌入式系统综合设计》 实验八:FFMPEG视频编码

一、实验目的 掌握使用算能平台进行视频编码的流程&#xff0c;包括开发主机环境与云平台的配置&#xff0c;视频编码程序的编写与理解&#xff0c;代码的编译、运行以及学习使用码流分析工具分析视频压缩码流等。 二、实验内容 搭建实验开发环境&#xff0c;编译并运行编码…

Spring Boot 3.2.0 虚拟线程初体验 (部分装配解析)

写在前面 spring boot 3 已经提供了对虚拟线程的支持。 虚拟线程和平台线程主要区别在于&#xff0c;虚拟线程在运行周期内不依赖操作系统线程&#xff1a;它们与硬件脱钩&#xff0c;因此被称为 “虚拟”。这种解耦是由 JVM 提供的抽象层赋予的。 虚拟线程的运行成本远低于平…

组合设计模式

package com.jmj.pattern.combination;/*** 菜单组件&#xff0c;属于抽象根节点*/ public abstract class MenuComponent {//菜单组件的名称protected String name;//菜单组件的层级protected int level;//添加子菜单public void add(MenuComponent menuComponent) {throw new…

12.Spring源码解析-其它标签解析

容易看出&#xff0c;Spring其实使用了一个Map了保存其映射关系&#xff0c;key就是命名空间的uri&#xff0c;value是NamespaceHandler对象或是Class完整名&#xff0c;如果发现是类名&#xff0c;那么用反射的方法进行初始化&#xff0c;如果是NamespaceHandler对象&#xff…

计算虚拟化之CPU——qemu解析

解析 qemu 的命令行&#xff0c;qemu 的命令行解析&#xff0c;就是下面这样一长串。 qemu_add_opts(&qemu_drive_opts);qemu_add_opts(&qemu_chardev_opts);qemu_add_opts(&qemu_device_opts);qemu_add_opts(&qemu_netdev_opts);qemu_add_opts(&qemu_nic_…

C语言枚举的作用是什么?

我在知乎上看到这个问题&#xff0c;一开始&#xff0c;也有一些疑惑&#xff0c;后面查了一些资料&#xff0c;对于这个问题&#xff0c;简单的说一下我的看法。 枚举有多大 枚举类型到底有多大&#xff0c;占多少空间呢&#xff1f;这个要具体情况具体分析&#xff0c;编译器…

【shell】多行重定向与免交互expect与ssh、scp的结合使用

目录 一、多行重定向 举例1&#xff1a;使用read命令接收用户的输入值会有交互过程 举例2&#xff1a;设置变量的值 举例3&#xff1a;创建用户密码 举例4&#xff1a;使用多行重定向写入文件中&#xff08;以repo文件举例&#xff09; 举例5&#xff1a;变量设定 二、免…

C++初阶模板

介绍&#xff1a; 我们先认识以下C中的模板。模板是一种编程技术&#xff0c;允许程序员编写与数据类型无关的代码&#xff0c;它是一种泛型编程的方式&#xff0c;可以用于创建可处理多种数据类型的函数或类&#xff0c;也就是说泛型编程就是编写与类型无关的通用代码&#xf…

多线程(补充知识)

STL库&#xff0c;智能指针和线程安全 STL中的容器是否是线程安全的? 不是. 原因是, STL 的设计初衷是将性能挖掘到极致, 而一旦涉及到加锁保证线程安全,会对性能造成巨大的影响. 而且对于不同的容器, 加锁方式的不同, 性能可能也不同(例如hash表的锁表和锁桶). 因此 STL 默认…

【LeetCode】每日一题 2023_11_25 二叉树中的伪回文路径(dfs,数组/位运算)

文章目录 刷题前唠嗑题目&#xff1a;二叉树中的伪回文路径题目描述代码与解题思路偷看大佬题解 结语 刷题前唠嗑 LeetCode&#xff1f;启动&#xff01;&#xff01;&#xff01; 这个月第一次周末早起~ 题目&#xff1a;二叉树中的伪回文路径 题目链接&#xff1a;1457. 二…

20世纪的葡萄酒有哪些创新?

葡萄酒是用酵母发酵的&#xff0c;直到20世纪中叶&#xff0c;这一过程都依赖于自然产生的酵母。这些发酵的结果往往不一致&#xff0c;而且由于发酵时间长&#xff0c;容易腐败。 酿酒业最重要的进步之一是在20世纪50、60年代引进了地中海的纯发酵菌种酿酒酵母&#xff0c;俗称…

你要的fiddler快捷键全部在这里了,学最全的快捷键,做最快的IT程序员

一、常用三个快捷键 ctrlX :清空所有记录 CtrlF&#xff1a;查找 F12&#xff1a;启动或者停止抓包 使用 QuickExec Fiddler2 成了网页调试必备的工具&#xff0c;抓包看数据。Fiddler2自带命令行控制。 fiddler 命令行快捷键&#xff1a;ctrl q &#xff0c;然后 输入 help…

Codeforces Round #911 (Div. 2)

A.Cover in Water 题意&#xff1a; 有一个 1 n 1 \times n 1n的水池&#xff0c;里面有些格子可以加水&#xff0c;有些格子是被堵上的&#xff0c;你可以进行以下两种操作&#xff1a; 1.往一个空的格子里加水 2.移除一个有水的格子中的水&#xff0c;并将这些水添加到另…

合并区间[中等]

一、题目 以数组intervals表示若干个区间的集合&#xff0c;其中单个区间为intervals[i] [starti, endi]。请你合并所有重叠的区间&#xff0c;并返回一个不重叠的区间数组&#xff0c;该数组需恰好覆盖输入中的所有区间。 示例 1&#xff1a; 输入&#xff1a;intervals […

【笔记】小白学习电路维修

学习视频&#xff08;b站&#xff09;&#xff1a;从0开始学电路 从0开始学电路维修 p1 黄色长方体元件P2 故障率最高的元件p3带芯铜丝线圈是什么区分电感和变压器接入电路分析&#xff1a; p4 交流和直流分界线整流桥接线整流桥故障判断 带色环的不一定是电阻 p1 黄色长方体元…

Windows10-用户账户控制、Windows远程桌面

Windows10用户账户控制怎么设置白名单 问题引出&#xff1a; 安装低版本搜狗输入法后经常弹出用户账户控制 解决方案&#xff1a; 全局模式&#xff1a; UAC控制最早出现在Windows Vista中&#xff0c;用户帐户控制&#xff08;UAC&#xff09;是一项旨在防止对您的计算机…

web:[ZJCTF 2019]NiZhuanSiWei1

题目 点进题目&#xff0c;网页显示如下&#xff0c;需要代码审计 $_GET["text"]和$_GET["file"]来获取传入的两个参数text和file。使用isset()函数来检查$text变量是否已设置并且不为null。如果设置了并且不为null&#xff0c;则执行下面的逻辑。在下面的…