专注于做有价值的技术原创

0%

Make 快速入门

1. Make

make 是 linux 系统的实用程序。它用于管理对于大型程序的自动编译任务,自动决定程序某一部分需要重新编译,并发出编译指令。虽然,我们最常见于 C 语言程序的编译。但是,make 不限于某一特定语言,凡是可以通过 shell 命令来运行编译器的语言都可以使用 make 。除此之外,你甚至可以用 make 描述任何构建任务,这些任务中,文件需要在其依赖的文件发生变动后自动更新。

2. Makefile

在使用 make 之前,你必须在当前目录下添加一个 Makefile 文件,它描述了文件之间的依赖(输入输出)关系,并提供更新文件的 Shell 命令。Makefile 文件名可以是 Makefile 也可以是 makefile ,推荐使用 Makefile

2.1 构成

简单的 Makefile 文件由若干如下格式的规则(rule)组成:

1
2
target: prerequisites
recipe
  • target : 通常是程序生成(输出)的一个或多个文件名,例如:可执行文件或目标文件;它也可以是要执行任务的名称,例如用于清理生成文件的 clean 任务。
  • prerequisites: 先决条件是用于生成 target 文件的输入文件或是完成 target 任务前需要先执行的任务 。一个 target 可以没有先决条件,也可以有一个或多个先决条件。
  • recipe: 中文翻译为菜谱,它是 make 用于生成 target 文件或完成 target 任务而执行一系列 shell 命令。这些命令可以放在同一行里,也可以每个命令占一行。值得注意的是,recipe 默认以制表符开头,而不是空格。

2.2 运行 make

在当前目录下创建一个 Makefile 文件, 命名为 Makefile 。将以下内容复制到新建的 Makefile 文件中。

注意: recipe 默认是以制表符开头,不是空格。如果复制到文件是空格,需要手动将空格改成制表符,即按键盘 tab 键。

1
2
3
4
5
6
7
8
9
10
.PHONY: help clean mk

help :
@echo "help info"

dist :
@mkdir dist

clean :
@rm -rf dist

.PHONY: help clean mk 用于告诉 make 指定 target 列表不是文件名。可以理解是纯粹的任务,而不生成文件。

在 Makefile 所在目录运行不带参数的 make 命令:

1
make

会启动 Makefile 文件中第一个 target ,本例是 help , make 将 Makefile 中第一个出现的 target 作为默认目标 。

要启动其他 target,需要在 make 命令后指定 target 名称。如下分别启动目标 dist 和 clean 。

1
2
make dist
make clean

*2.3 扩展:对比 npm scripts

对于前端的同学,最熟悉的构建方式莫过于 npm package.json 中 scripts 构建命令。在不使用项目中安装的包的情况下,即项目目录下 node_modules 里的包,完全可以使用 Makefile 来完成一些构建任务,make 的优势在于更好管理相互依赖的构建任务。

3. 变量

make 中的变量本质是一种宏替换,用于简化和维护重复出现的字符串和字符串列表。既可以出现在目标,先决条件,也可以出现在“菜谱”的 shell 命令中;可以是命令本身,也可以是命令的选项,或者输入输出文件;甚至也可以出现在另一个变量的引用中(计算变量)。

3.1 定义变量

make 变量定义和 shell 变量的定义非常相似。不同的是 make 变量的名称可以是任何不包含 :, #, = 和空字符的字符序列,并且等号两边可以有空格(shell 定义变量的等号两个不允许出现空格)。

如下定义一个变量 objs ,用于表示 c 语言编译器输出的一系列目标文件。

1
objs = main.o model.o view.o controller.o

3.2 使用变量

make 变量支持和 shell 变量一样,在变量标识符前加美元符 $ 来引用,因此,如果在”菜谱”中使用 shell 变量,需要使用双美元符 $$ 作为前缀加以区分;但更推荐的使用方式是使用美元符后跟一对圆括号的方式,例如使用上文创建的变量 objs ,可以这样 $(objs)

1
2
build: $(objs)
cc -o app $(objs)

4. 函数

make 中的函数用于处理 Makefile 文件中的文本,例如:计算操作的文件列表,“菜谱”中使用的命令等。

4.1 函数调用

函数调用类似于变量引用,它可以出现在任何变量引用可以出现的地方。调用方式如下:

1
$(function arguments)

或者:

1
${function arguments}

其中,function 是函数名称,arguments 是函数的参数。函数名与参数之间用空格或制表符隔开,多个参数之间用逗号 , 隔开。

4.2 用于字符串替换和分析的函数

4.2.1 $(subst from,to,text)

text 上执行文本替换,将出现的所有 from 替换成 to

Makefile:

1
2
subst:
@echo $(subst o,O,hello world)

make:

1
2
$ make subst
hellO wOrld

4.2.2 $(patsubst pattern,replacement,text)

text 中寻找空格分隔的单词,如果单词匹配 pattern , 就将匹配的单词替换成 replacementpatternreplacement 都可以包含通配符 %,匹配任意数量的任意字符。当 patternreplacement 同时包含通配符 %,则将 replacement% 替换成与 pattern% 匹配的文本。

Makefile:

1
2
patsubst:
@echo $(patsubst %.c,%.o,hello.c.c world.c)

make:

1
2
$ make patsubst
hello.c.o world.o

4.2.3 $(strip string)

移除字符串 string 首尾空格,并且将字符串中的多个空格替换成一个空格。

Makefile:

1
2
strip:
@echo $(strip hello world ! )

make:

1
2
$ make strip
hello world !

4.2.4 $(sort list)

按英文字母表顺序对列表 list 中的单词进行排序,删除重复的单词。输出是由单个空格分隔的单词列表。

Makefile:

1
2
sort:
@echo $(sort dog cat animal fox bear fox elephant)

make:

1
2
$ make sort
animal bear cat dog elephant fox

4.2.5 $(words text)

返回 text 中单词数量。

Makefile:

1
2
words:
@echo $(words dog cat animal fox bear fox elephant)

make:

1
2
$ make words
7

4.2.6 $(word n,text)

返回 text 中第 n 个单词,n 从 1 开始计数。

Makefile:

1
2
3
animals = bear cat dog elephant fox
word:
@echo $(word $(words $(animals)), $(animals))

make:

1
2
$ make word
fox

5. 隐式规则

某些重新生成目标文件的方式非常常用。例如,使用 C 编译器 cc.c 源文件编译生成 .o 目标文件。 隐式规则将告诉 make 如何使用常用的技术,让你在使用时不必给出全部细节,简化书写。例如,make 为 C 语言编译提供一个隐式规则。文件名决定将应用哪个隐式规则。比如,C 编译通常输入 .c 文件,输出 .o 文件。 因此,当看到文件名结尾符合这种组合时,make 将隐式规则应用于 C 编译。

Makefile:

1
2
3
4
5
6
7
8
app : main.o utils.o
cc -o app main.o utils.o

main.o : main.c utils.h
cc -c main.c

utils.o : utils.c
cc -c utils.c

可以去掉编译命令,简写为:

1
2
3
4
5
6
app : main.o utils.o
cc -o app main.o utils.o

main.o : main.c utils.h

utils.o : utils.c

make:

1
2
3
4
$ make
cc -c -o main.o main.c
cc -c -o utils.o utils.c
cc -o app main.o utils.o

参考

  • [1] gnu.org: GNU make
青笔 wechat
我是一条小青蛇 我有很多小秘密
  • 本文作者: 青笔
  • 本文链接: http://www.qingbii.com/2019/10/05/make/
  • 版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!