1. Make
make 是 linux 系统的实用程序。它用于管理对于大型程序的自动编译任务,自动决定程序某一部分需要重新编译,并发出编译指令。虽然,我们最常见于 C 语言程序的编译。但是,make 不限于某一特定语言,凡是可以通过 shell 命令来运行编译器的语言都可以使用 make 。除此之外,你甚至可以用 make 描述任何构建任务,这些任务中,文件需要在其依赖的文件发生变动后自动更新。
2. Makefile
在使用 make 之前,你必须在当前目录下添加一个 Makefile 文件,它描述了文件之间的依赖(输入输出)关系,并提供更新文件的 Shell 命令。Makefile
文件名可以是 Makefile
也可以是 makefile
,推荐使用 Makefile
。
2.1 构成
简单的 Makefile
文件由若干如下格式的规则(rule
)组成:
target: prerequisites
recipe
- target : 通常是程序生成(输出)的一个或多个文件名,例如:可执行文件或目标文件;它也可以是要执行任务的名称,例如用于清理生成文件的
clean
任务。 - prerequisites: 先决条件是用于生成 target 文件的输入文件或是完成 target 任务前需要先执行的任务 。一个 target 可以没有先决条件,也可以有一个或多个先决条件。
- recipe: 中文翻译为菜谱,它是 make 用于生成 target 文件或完成 target 任务而执行一系列 shell 命令。这些命令可以放在同一行里,也可以每个命令占一行。值得注意的是,recipe 默认以制表符开头,而不是空格。
2.2 运行 make
在当前目录下创建一个 Makefile 文件, 命名为 Makefile 。将以下内容复制到新建的 Makefile 文件中。
注意: recipe 默认是以制表符开头,不是空格。如果复制到文件是空格,需要手动将空格改成制表符,即按键盘 tab 键。
代码语言:javascript复制.PHONY: help clean mk
help :
@echo "help info"
dist :
@mkdir dist
clean :
@rm -rf dist
.PHONY: help clean mk
用于告诉 make 指定 target 列表不是文件名。可以理解是纯粹的任务,而不生成文件。
在 Makefile 所在目录运行不带参数的 make 命令:
代码语言:javascript复制make
会启动 Makefile 文件中第一个 target ,本例是 help , make 将 Makefile 中第一个出现的 target 作为默认目标 。
要启动其他 target,需要在 make 命令后指定 target 名称。如下分别启动目标 dist 和 clean 。
代码语言:javascript复制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 语言编译器输出的一系列目标文件。
代码语言:javascript复制objs = main.o model.o view.o controller.o
3.2 使用变量
make 变量支持和 shell 变量一样,在变量标识符前加美元符 $
来引用,因此,如果在”菜谱”中使用 shell 变量,需要使用双美元符 $$
作为前缀加以区分;但更推荐的使用方式是使用美元符后跟一对圆括号的方式,例如使用上文创建的变量 objs ,可以这样 $(objs)
。
build: $(objs)
cc -o app $(objs)
4. 函数
make 中的函数用于处理 Makefile 文件中的文本,例如:计算操作的文件列表,“菜谱”中使用的命令等。
4.1 函数调用
函数调用类似于变量引用,它可以出现在任何变量引用可以出现的地方。调用方式如下:
代码语言:javascript复制$(function arguments)
或者:
代码语言:javascript复制${function arguments}
其中,function 是函数名称,arguments 是函数的参数。函数名与参数之间用空格或制表符隔开,多个参数之间用逗号 ,
隔开。
4.2 用于字符串替换和分析的函数
4.2.1 $(subst from,to,text)
在 text
上执行文本替换,将出现的所有 from
替换成 to
。
Makefile:
代码语言:javascript复制subst:
@echo $(subst o,O,hello world)
make:
代码语言:javascript复制$ make subst
hellO wOrld
4.2.2 $(patsubst pattern,replacement,text)
在 text
中寻找空格分隔的单词,如果单词匹配 pattern
, 就将匹配的单词替换成 replacement
。pattern
和 replacement
都可以包含通配符 %
,匹配任意数量的任意字符。当 pattern
和 replacement
同时包含通配符 %
,则将 replacement
中 %
替换成与 pattern
中 %
匹配的文本。
Makefile:
代码语言:javascript复制patsubst:
@echo $(patsubst %.c,%.o,hello.c.c world.c)
make:
代码语言:javascript复制$ make patsubst
hello.c.o world.o
4.2.3 $(strip string)
移除字符串 string
首尾空格,并且将字符串中的多个空格替换成一个空格。
Makefile:
代码语言:javascript复制strip:
@echo $(strip hello world ! )
make:
代码语言:javascript复制$ make strip
hello world !
4.2.4 $(sort list)
按英文字母表顺序对列表 list
中的单词进行排序,删除重复的单词。输出是由单个空格分隔的单词列表。
Makefile:
代码语言:javascript复制sort:
@echo $(sort dog cat animal fox bear fox elephant)
make:
代码语言:javascript复制$ make sort
animal bear cat dog elephant fox
4.2.5 $(words text)
返回 text
中单词数量。
Makefile:
代码语言:javascript复制words:
@echo $(words dog cat animal fox bear fox elephant)
make:
代码语言:javascript复制$ make words
7
4.2.6 $(word n,text)
返回 text
中第 n
个单词,n
从 1 开始计数。
Makefile:
代码语言:javascript复制animals = bear cat dog elephant fox
word:
@echo $(word $(words $(animals)), $(animals))
make:
代码语言:javascript复制$ make word
fox
5. 隐式规则
某些重新生成目标文件的方式非常常用。例如,使用 C 编译器 cc
从 .c
源文件编译生成 .o
目标文件。 隐式规则将告诉 make
如何使用常用的技术,让你在使用时不必给出全部细节,简化书写。例如,make 为 C 语言编译提供一个隐式规则。文件名决定将应用哪个隐式规则。比如,C 编译通常输入 .c
文件,输出 .o
文件。 因此,当看到文件名结尾符合这种组合时,make 将隐式规则应用于 C 编译。
Makefile:
代码语言:javascript复制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
可以去掉编译命令,简写为:
代码语言:javascript复制app : main.o utils.o
cc -o app main.o utils.o
main.o : main.c utils.h
utils.o : utils.c
make:
代码语言:javascript复制$ 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