Make 快速入门

2019-10-23 17:08:39 浏览数 (1)

1. Make

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

2. Makefile

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

2.1 构成

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

代码语言:javascript复制
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)

代码语言:javascript复制
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 , 就将匹配的单词替换成 replacementpatternreplacement 都可以包含通配符 %,匹配任意数量的任意字符。当 patternreplacement 同时包含通配符 %,则将 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

0 人点赞