【Linux】Makefile:让编译自动化

当项目只有一个 hello.c 时,手动敲 gcc 命令没有负担。但当源文件增加到几十个、上百个,分布在不同的目录里,而且有复杂的依赖关系时,每次手动编译就变成了效率黑洞。

Makefile 定义了源文件之间的依赖关系和编译规则,然后通过 make 命令执行自动化构建。它最核心的用途是只重新编译那些被修改过的文件,而不是整个项目重新来过。

1. 一个最简 Makefile

makefile

复制代码
myproc: myproc.c
	gcc -o myproc myproc.c

.PHONY: clean
clean:
	rm -f myproc

这里有两个目标(target):

  • myproc:依赖 myproc.c。依赖关系下面缩进的那行是依赖方法------具体要执行的命令。

  • clean:不依赖任何文件,执行 rm -f 清理生成的可执行文件。

make 默认执行文件中的第一个目标(这里是 myproc)。执行 make clean 则显式指定执行 clean 目标。

clean 被声明为 .PHONY,意思是它是一个伪目标 ,不代表一个真实的输出文件。伪目标的特性是"总是被执行"------无论 clean 文件是否已经存在,make clean 都会无条件运行。如果不加 .PHONY 且当前目录下恰好有一个叫 clean 的文件,make clean 可能会因为"依赖文件已最新"而拒绝执行。

2. 依赖推导:make 是如何工作的

make 的机制可以概括为"逐层检查与构建"。它从第一个目标开始,检查它依赖的文件是否存在、是否比目标更新。如果依赖不存在,则递归查找能生成该依赖的规则。如果依赖比目标新(说明源码修改过),则执行生成命令。

假如有这样一个文件体系:

makefile

复制代码
myproc: myproc.o
	gcc myproc.o -o myproc
myproc.o: myproc.s
	gcc -c myproc.s -o myproc.o
myproc.s: myproc.i
	gcc -S myproc.i -o myproc.s
myproc.i: myproc.c
	gcc -E myproc.c -o myproc.i

make 在构建myproc时会沿着依赖链一直往回推:

  1. 发现需要 myproc.o,但目录下没有,于是找生成 myproc.o 的规则。

  2. 发现需要 myproc.s,又没有,继续回溯。

  3. 最终追溯到底层的 myproc.c,然后按顺序执行:.c → .i → .s → .o → myproc

这样,如果只修改了一个源文件,make 只重编译受影响的目标,其它已经是最新的不会重复编译。

3. 变量与模式规则

重复写文件名容易出错,也不好维护。make 支持定义变量:

makefile

复制代码
BIN = myproc
CC = gcc
SRC = $(wildcard *.c)
OBJ = $(SRC:.c=.o)

$(BIN): $(OBJ)
	$(CC) -o $@ $^

%.o: %.c
	$(CC) -c $@ $<

.PHONY: clean
clean:
	rm -f $(OBJ) $(BIN)

几个关键点:

  • $(wildcard *.c):获取当前目录下所有 .c 文件名。

  • $(SRC:.c=.o):把 .c 后缀替换成 .o

  • $@:当前规则的目标文件名。

  • $^:当前规则的所有依赖文件名。

  • $<:当前规则的第一个依赖文件名。

  • %.o: %.c 是一个模式规则,对所有 .o 目标匹配同名的 .c 依赖。

@前缀放在命令前的作用是关闭命令本身的回显,让输出更干净,只显示像"linking..."、"compiling..."这样的关键信息。

这套写法写好以后,当源文件新增时 Makefile 通常不需要改动,这就是自动化构建的真正价值。

相关推荐
阿里云大数据AI技术14 小时前
阿里云 EMR AI 助手正式发布:从问答工具到全栈智能运维助手
运维·人工智能
orion571 天前
Missing Semester Class1:course overview and introduction of shell
linux
SkyWalking中文站1 天前
认识 Horizon UI · 6/17:Trace 探索器
运维·监控·自动化运维
用户120487221612 天前
Linux驱动编译与加载
linux·嵌入式
火车叼位2 天前
写给初级开发者:SSL、SSH、HTTPS 与证书体系全解析
运维
用户805533698032 天前
Input 子系统架构:Core、Handler、Driver 三层是怎么协作的
linux·嵌入式
用户805533698032 天前
RK-Forge外设系列开篇 - 把板子从「能启动」变成「能用」:Ethernet/SPI/MMC 三个纯接线外设
linux·github·嵌入式
小猿姐2 天前
唯品会大规模数据库云原生实践:基于 KubeBlocks 管理数千实例的统一运维之路
运维·elasticsearch·云原生
七歌杜金房2 天前
我终于又有了自己的 Linux 电脑
linux·debian·mac
SkyWalking中文站3 天前
认识 Horizon UI · 5/17:3D 基础设施地图
运维·监控·自动化运维