GNU make 与 Makefile

前言

make 是一个工具程序,通过读取 Makefile 文件,实现自动化软件构建。虽然现代软件开发中,集成开发环境已经取代了 make,但在 Unix 环境中,make 仍然被广泛用来协助软件开发。make 不仅仅用来编译 C/C++ 程序,还可以用来实现任何输入文件和结果文件的转化。

Makefile 语法

Makefile 文件由一组规则构成,每个规则有如下形式:

targets : prerequisities
	recipe
	recipe
	recipe
  • targets:目标是文件名或操作名(也称为“伪目标”,phony target)。目标通常是一个,多个目标之间用空格分隔。
  • prerequisities:前置条件定义了构建目标文件的依赖。多个依赖文件(也可以是伪目标)用空格分隔。只要有一个前置文件不存在,或者有过更新(前置文件的最后修改时间比目标的时间戳新),目标就需要重新构建。如果前置条件不存在,会执行 make <prerequisities>。
  • recipe:每一行配方的起首是 tab 键,而不是空格。

如果存在 targets 同名的文件,make <targets> 不会执行。这时需要加上 .PHONY,伪目标会在每次编译时重新执行一遍。

.PHONY: clean
clean:
	rm *.o temp

如果 make 执行时没有指定目标,则默认执行 Makefile 中的第一个目标

注释

# 在 Makefile 中表示注释。

变量

可以在 Makefile 中使用变量。引用变量的方法是:$(变量名)

files = file1 file2
some_file: $(files)
	echo "Look at this variable: " $(files)

赋值运算符可以是:=,=:,?=,

  • = 只有在命令使用到该变量时,才执行赋值操作。在此之前,变量会原封不动地记录赋值语句右边的表达式。
  • =: 赋值语句立刻执行。如果该赋值语句右边表达式中含有变量,则该变量只有定义在赋值语句之前才会被应用。
  • ?= 赋值前会检查变量是否已经被赋值。如果已赋值则不执行此次赋值,否则执行

指令

include

include 指令会使 make 程序停在当前 makefile 文件读取位置,而先去读取指定文件,再继续读。
-include 同 include,但文件不存在时不会报错

include Makefile.common

控制流指令

使用条件判断,可以让make根据运行时的不同情况选择不同的执行分支。下面的例子,判断 $(CC) 变量是否 gcc ,如果是的话,则使用GNU函数编译目标。

libs_for_gcc = -lgnu
normal_libs =

ifeq ($(CC),gcc)
    libs=$(libs_for_gcc)
else
    libs=$(normal_libs)
endif

foo: $(objects)
    $(CC) -o foo $(objects) $(libs)

ifeq 如果相等
ifneq 如果不相等

配方

显示命令

make 默认会把要执行的命令在执行前输出到屏幕上。但如果在命令行前用 @,则不会打印。比如下面的例子,只打印 >> writing assets,而不会打印 echo。

.PHONY: assets
assets:
	@echo ">> writing assets"

函数

字符操作函数

# firstword 打印第一个单词
$(firstword foo bar)

# word <N> 打印第 N 个单词
$(word 2, foo bar baz)

# wildcard <FILENAMES> 打印所有匹配的文件名,用空格隔开
$(wildcard *.md)

shell 函数

shell 函数执行 shell 后面跟着的参数,并把执行结果作为返回。

go = $(shell go version)

.PHONY: all
all:
	echo $(go)

参考文档

[1] 阮一峰:Make 命令教程
[2] GNU Make Manual
[3] 跟我一起写 Makefile

posted @ 2019-11-22 21:25  黄挤挤  阅读(405)  评论(0编辑  收藏  举报