(给Linux爱好者加星标,提升Linux技能)
转自:ken爱编程
1. Make
make 是 linux 系统的实用程序。它用于管理对于大型程序的自动编译任务,自动决定程序某一部分需要重新编译,并发出编译指令。虽然,我们最常见于 C 语言程序的编译。但是,make 不限于某一特定语言,凡是可以通过 shell 命令来运行编译器的语言都可以使用 make 。除此之外,你甚至可以用 make 描述任何构建任务,这些任务中,文件需要在其依赖的文件发生变动后自动更新。
2. Makefile
在使用 make 之前,你必须在当前目录下添加一个 Makefile 文件,它描述了文件之间的依赖(输入输出)关系,并提供更新文件的 Shell 命令。Makefile文件名可以是Makefile也可以是makefile,推荐使用Makefile。
2.1 构成
简单的Makefile文件由若干如下格式的规则(rule)组成:
target : 通常是程序生成(输出)的一个或多个文件名,例如:可执行文件或目标文件;它也可以是要执行任务的名称,例如用于清理生成文件的clean任务。
prerequisites: 先决条件是用于生成 target 文件的输入文件或是完成 target 任务前需要先执行的任务 。一个 target 可以没有先决条件,也可以有一个或多个先决条件。
recipe: 中文翻译为菜谱,它是 make 用于生成 target 文件或完成 target 任务而执行一系列 shell 命令。这些命令可以放在同一行里,也可以每个命令占一行。值得注意的是,recipe 默认以制表符开头,而不是空格。
2.2 运行 make
在当前目录下创建一个 Makefile 文件, 命名为 Makefile 。将以下内容复制到新建的 Makefile 文件中。
注意:recipe 默认是以制表符开头,不是空格。如果复制到文件是空格,需要手动将空格改成制表符,即按键盘 tab 键。
.PHONY: help clean mk用于告诉 make 指定 target 列表不是文件名。可以理解是纯粹的任务,而不生成文件。
在 Makefile 所在目录运行不带参数的 make 命令:
会启动 Makefile 文件中第一个 target ,本例是 help , make 将 Makefile 中第一个出现的 target 作为默认目标 。
要启动其他 target,需要在 make 命令后指定 target 名称。如下分别启动目标 dist 和 clean 。
2.3 扩展:对比 npm scripts
对于前端的同学,最熟悉的构建方式莫过于 npm package.json 中 scripts 构建命令。在不使用项目中安装的包的情况下,即项目目录下 node_modules 里的包,完全可以使用 Makefile 来完成一些构建任务,make 的优势在于更好管理相互依赖的构建任务。
3. 变量
make 中的变量本质是一种宏替换,用于简化和维护重复出现的字符串和字符串列表。既可以出现在目标,先决条件,也可以出现在“菜谱”的 shell 命令中;可以是命令本身,也可以是命令的选项,或者输入输出文件;甚至也可以出现在另一个变量的引用中(计算变量)。
3.1 定义变量
make 变量定义和 shell 变量的定义非常相似。不同的是 make 变量的名称可以是任何不包含:,#,=和空字符的字符序列,并且等号两边可以有空格(shell 定义变量的等号两个不允许出现空格)。
如下定义一个变量 objs ,用于表示 c 语言编译器输出的一系列目标文件。
3.2 使用变量
make 变量支持和 shell 变量一样,在变量标识符前加美元符$来引用,因此,如果在"菜谱"中使用 shell 变量,需要使用双美元符$$作为前缀加以区分;但更推荐的使用方式是使用美元符后跟一对圆括号的方式,例如使用上文创建的变量 objs ,可以这样$(objs)。
4. 函数
make 中的函数用于处理 Makefile 文件中的文本,例如:计算操作的文件列表,“菜谱”中使用的命令等。
4.1 函数调用
函数调用类似于变量引用,它可以出现在任何变量引用可以出现的地方。调用方式如下:
或者:
其中,function 是函数名称,arguments 是函数的参数。函数名与参数之间用空格或制表符隔开,多个参数之间用逗号 , 隔开。
4.2 用于字符串替换和分析的函数
4.2.1 $(subst from,to,text)
在text上执行文本替换,将出现的所有from替换成to。
Makefile:
make:
4.2.2 $(patsubst pattern,replacement,text)
在text中寻找空格分隔的单词,如果单词匹配pattern, 就将匹配的单词替换成replacement。pattern和replacement都可以包含通配符%,匹配任意数量的任意字符。当pattern和replacement同时包含通配符%,则将replacement中%替换成与pattern中%匹配的文本。
Makefile:
make:
4.2.3 $(strip string)
移除字符串string首尾空格,并且将字符串中的多个空格替换成一个空格。
Makefile:
make:
4.2.4 $(sort list)
按英文字母表顺序对列表list中的单词进行排序,删除重复的单词。输出是由单个空格分隔的单词列表。
Makefile:
make:
4.2.5 $(words text)
返回text中单词数量。
Makefile:
make:
4.2.6 $(word n,text)
返回text中第n个单词,n从 1 开始计数。
Makefile:
make:
隐式规则
某些重新生成目标文件的方式非常常用。例如,使用 C 编译器cc从.c源文件编译生成.o目标文件。隐式规则将告诉make如何使用常用的技术,让你在使用时不必给出全部细节,简化书写。例如,make 为 C 语言编译提供一个隐式规则。文件名决定将应用哪个隐式规则。比如,C 编译通常输入.c文件,输出.o文件。因此,当看到文件名结尾符合这种组合时,make 将隐式规则应用于 C 编译。
Makefile:
可以去掉编译命令,简写为:
make:
领取专属 10元无门槛券
私享最新 技术干货