makefile文件详解

1. make

  • 编译:将源代码文件翻译成处理器可执行的二进制文件的过程,这个过程的时间区间称为编译时
  • 构建:指定多个编译过程的先后顺序

make命令是常用的构建工具,诞生于1977年,主要用于C/C++项目工程的构建,make命令的适用场景:只要某一个或多个文件发生变动,就要重新构建项目

2. Makefile

make的语义是制作,制作出某样工件,工件可以是各种可执行文件、中间文件、镜像等,通过Makefile文件描叙构建工件的过程,构建过程由一个或多个规则组成

  • make 命令默认使用工程项目目录下的Makefile文件,可以通过 -f --file 选项指定其位置的Makefile文件

  • 规则(rules)

<target>: <precondition> ...
|table|<command>
...
规则文法:
1. 冒号前面的 <target> 是构建的目标,可以是所构建对象的文件名,或者构建过程达成的目标名
2. 冒号后面的 <precondition> ... 是前置条件,判断目标是否重新构建,重新构建的条件为:只要有一个前置文件不存在或者有过更新(前置文件的mtime比目标的mtime新),跳过已经编译但未修改的文件,降低总编译时间
3. |table| 是一个缩进,必须以该字符起首
4. <command> 是命令,达成该目标需要执行的子命令
5. |table| <command> 可以有多个,是具体的执行过程,<prerequest> |table|<command> 至少存在一个
2.1 目标 -- target

一个目标构成一条规则

  • 目标可以是文件名
# 会在目录下创建 a.txt b.txt文件,b.txt文件先于a.txt文件创建,若b.txt文件的mtime时间戳比a.txt文件的时间戳新则重新执行a.txt的构建过程
# 通过判断当前工程目录下的文件是否存在,判断是否重新执行目标的构建过程
a.txt: b.txt
	cat b.txt > a.txt

b.txt:
	echo "make comand" > b.txt
  • 目标可以是某个操作名,若工程目录存在与操作同名的文件,就会跳过该目标的执行过程,可以通过 .PHONY声明为伪目标,不会检查目标名对应的文件是否存在
# 若当前工程目录下已存在clean文件,则跳过,可以通过 .PHONY指定该目标是一个非文件目标
.PHONY: clean 
clean:
	rm a.txt b.txt
2.2 前置条件 -- precondition

一组以空格隔开的文件名,用于判断目标是否重新构建,如果至少有一个文件不存在或者文件的最后修改时间比目标文件的最后修改时间新,则重新构建该目录

# 会首先依次构建前置条件中的目标文件,最后构建目标文件,反复构建,只要前置条件中的文件最后修改时间戳比目标文件小则跳过目标文件的构建过程
result.txt: file1.txt file2.txt file3.txt
	cat file1.txt file2.txt file3.txt > result.txt

file1.txt:
	echo "file1" > file1.txt

file2.txt:
	echo "file2" > file2.txt

file3.txt:
	echo "file3" > file3.txt
2. 3 命令 -- command

描叙目标是如何创建出来的,由一个或多个shell命令描叙,是构建目标的具体指令,每一行命令在一个进程中并行执行

  • 每个构建命令在各自的shell进程中执行,通过 .ONESHELL指定目标的命令集合都在同一个shell进程中串行
# 看不到变量NAME的值,需要使用$进行$符号的转义,输出不了变量的值
var-get:
	export NAME=cloud
	echo $$NAME
# 写在一行的方式,以分号隔开shell指令
var-get:
	export NAME=cloud;echo $$NAME
# export 与 echo 指令都在同一个shell进程中运行
.ONESHELL:
var-get:
	export NAME=cloud
	echo $$NAME

  • 可以通过 .RECIPEPREFIX 修改命令的前缀,默认为Table字符
result.txt: file1.txt file2.txt file3.txt
	cat file1.txt file2.txt file3.txt > result.txt

# 之后的构建目标的指令列表由 > 开头
.RECIPEPREFIX = >
file1.txt:
> echo "file1" > file1.txt

file2.txt:
> echo "file2" > file2.txt

file3.txt:
> echo "file3" > file3.txt

3. 文法

向make命令描叙字符文字的语法与语义

3.1 注释

以 # 开头,到行尾结束,Bash 与 Python语言 的注释也一样

# export 与 echo 指令都在同一个shell进程中运行
.ONESHELL:
var-get:
	export NAME=cloud	# 设置一个环境变量
	echo $$NAME			# 输出环境变量的值
3. 2 回声

echo -- 执行make命令会打印出每个目标的命令然后再执行,在命令的最开头添加@关闭回声

# export 与 echo 指令都在同一个shell进程中运行
.ONESHELL:
var-get:
	@export NAME=cloud	
	echo $$NAME			

3.2 通配符

选定一组文件名符合通配符的文件,文法规则和bash一致

  • 单个字符
  • 任意长度任意字符*
  • 字符集中单个字符 []
# 删除所有以 .0 结尾的文件
clean:
	rm -f *.o	
3.3 模式匹配

选定文件名相同,但文件类型不同的文件

# 所有 .c 结尾的文件通过命令生成同名的.o文件
%.o: %.c
3. 4 变量与赋值符

声明变量并存储值,在之后代码中访问并引用变量中值

  • 赋值 = := ?= +=
VARIABLE = value
# 在执行时扩展,允许递归扩展。

VARIABLE := value
# 在定义时扩展。

VARIABLE ?= value
# 只有在该变量为空时才设置值。

VARIABLE += value
# 将值追加到变量的尾端。
  • 访问值

通过 $()访问,访问bash变量需要加上$进行转义

# 访问定义的变量与bash变量
name = dream_flish
get-variable:
	@echo $(name)
	@echo $$HOME
  • 内置变量

当前编辑器 CC ,当前使用的make工具 $(MAKE)

# 访问定义的变量与bash变量
name = dream_flish
get-variable:
	@echo "$(CC) $(MAKE)" 
	@echo $$HOME
  • 自动变量

变量的值与当前的规则的目标与前置条件相关

1. $@ 当前目标
2. $< 第一个前置条件
3. $? 比目标文件mtime时间戳更新的前置条件文件,多个以空格隔开
4. $^ 所有前置条件
5. $* 匹配 % 部分
6. $(@D) 目标当前目录名
7. $(@F) 目标当前文件名
8. $(<D) 第一个前置条件的目录名
9. $(<F) 第一个前置条件的文件名
a.txt: b.txt c.txt
	echo $@ # 等价 echo a.txt
	cat $^ > a.txt # 等价 cat b.txt c.txt > a.txt

b.txt:
	echo "b" > b.txt

c.txt:
	echo "c" > c.txt
3. 5 判断与循环

流程控制,满足条件时执行一段代码,不满足时执行另一段代码,利用Bash代码实现循环

BOY_NAMES = ALEX BEI XIXI
GIRL_NAMES = LANGLANG XUEXUE XIAOXIAO
CONDATION = BOY

# 等值判断
ifeq ($(CONDATION), BOY)
	NAMES=$(BOY_NAMES)
else
	NAMES=$(GIRL_NAMES)
endif

# 访问变量中的值
all:
	for name in $(NAMES); do echo $$name; done

4. 函数

完成某项功能的指令集合,调用文法: $(函数名, 参数 ... )

  • shell函数 -- 执行shell命令
files := $(shell seq 1 10)

all:
	echo $(files)
  • wildcard 通配符函数 -- 替换Bash的通配符
# 列出 src目录下所有以.tst后缀结尾的文件
files := $(wildcard src/*.txt)
all:
	echo $(files)
  • subst函数 -- 文本替换
# 小写的i换成大写的I
result := $(subst i,I,i Love C++)
all:
	echo $(result)

5. 编译 C++代码

  1. 在工程目录创建src目录,再在src目录下创建main.cpp文件,写入以下内容
#include <iostream>

int main() {
    std::cout << "I Love C++\n";
}
  1. 在工程目录下创建Makefile文件,写入以下规则
TARGET_DIR = bin
all:
	[ ! -d $)TARGET_DIR ] && mkdir -p $(TARGET_DIR); \
	c++ src/main.cpp -o $(TARGET_DIR)/main


clean:
	rm -f bin/*
  1. 在工程目录执行构建命令并执行编译后的结果
# 编译,没有目标则默认找到的第一个目标
make

# 执行,会输出 I Love C++
./bin/main

# 清理,make命令中输出目标名,多个目标以空格隔开
make clean
posted @   梦_鱼  阅读(184)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· 25岁的心里话
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示