Makefile详解

本文深入讲解Makefile的基础概念、基本原则及语法,包括编译、链接、静态库与动态库的区别,详细介绍了Makefile的规则、命令及变量定义,同时探讨了make的执行原理和常见命令行参数。

Makefile详解


其中的内容大多来自《GNUmake中文手册》3.8版徐海兵译,有小部分是我的理解:


在开始之前,我们要了解一些基本概念:

编译:把高级语言转化成.o文件

链接:把多个.o文件和库文件链接成一个可执行程序,“ld”命令

静态库:Linux下的静态库是.a文件,“ar”命令,程序可以把依赖的.a文件加到自己的程序中,这样,程序的运行就不需要外部的库函数了,但缺点是体积比较大,且依赖的任何一个静态库文件发生变化就要重新编译源程序。

动态库:Linux下的动态库是.so文件,程序无需连接时就把依赖的库文件连接进自己的程序中,这样体积小,但程序的运行需要依赖外部库函数。



make的基本原则:

1.如果所有的源文件没有被编译过,则对各个源文件编译并连接,生成最后的可执行程序;

2.每一个在上次执行make之后修改过的源代码文件在本次执行make时将会被重新编译;

3.头文件在上一次执行make之后被修改,则所有包含此头文件的源文件在本次执行make时将会被重新编译。



Makefile的基本语法:

TARGETPREREQUISITES

COMMAND


TARGET规则的目标。通常是程序中间或者最后需要生成的文件名。可以是.o文件、也可以是最后的可执行程序的文件名。另外,目标也可以是一个make执行的动作的名称,如目标“clean”,称这样的目标是“伪目标”。

PREREQUISITES规则的依赖。生成规则目标所需要的文件名列表。通常一个目标依赖于一个或者多个文件。

COMMAND规则的命令行。任意的shell命令或者可在shell下执行的程序。


注意:每一个COMMAND必须以[Tab]字符开始,[Tab]字符告诉make此行是一个命令行。这是书写Makefile最容易产生的错误。



make原理:

当在shell提示符下输入“make”命令以后。make读取当前目录下的Makefile文件,并将

Makefile文件中的第一个目标作为其“终极目标”,开始处理第一个规则(终极目标所在的规则)。由于该目标依赖于其他目标,所以接下来一层层地去寻找终极目标的依赖文件所在的规则并执行。当终极目标的规则被完全的展开以后,make将从最后一个被展开的规则处开始执行,之后处理倒数第二个规则,......依次回退。最后一步执行的就是终极目标所在的规则。整个过程就类似于C语言中的递归实现一样。同样,如果在shell提示符下输入“makexxx”命令后,make会在Makefile文件中查找xxx目标,然后再递归的查找依赖的目标执行。因此如果xxx目标没有依赖另一个目标,那个目标的命令就不会被执行,所以我们“make”的时候,目标“clean”的动作“rm”就没有被执行。


例:

#samplemakefile
#定义变量
object= hello1.o hello2.o hello3.o
hello:$(object)
gcc-o hello $(object)
hello1.o:hello1.chello.h
gcc-c hello1.c
hello2.o:hello2.chello.h
gcc-c hello2.c
hello3.o:hello3.chello.h
gcc-c hello3.c
#伪目标
.PHONY:clean
clean:
rm$(object) hello

上例中的“.PHONY: clean ”表示将“clean”目标声明为伪目标,一般情况下这句可以省略,但如果当前工作目录下存在文件“clean”时就不能省略了,否则,我们执行“makeclean”时,规则没有依赖文件,所以目标被认为是最新的而不去执行规则作定义的命令,因此命令“rm”将不会被执行!




几点注意:

1. 

在书写时,一个较长行可以使用反斜线(\)分解为多行,注意:反斜线之后不能有空格。

2.

Makefile中“#”字符后的内容被作为是注释(和shell脚本一样)。如果一行的第一个非空字符为“#”,那么此行 为注释行。注释行的结尾如果存在反斜线(\)那么下一行也被作为注释行。

注意:推荐在书写Makefile时将注释作为独立的一 行!

当在Makefile中需要使用字符“#”时,可以使用“\#”来实现,其表示将“#”作为一字符而不是注释的开始标志

3.

如果命令前有“-”则表示如果该命令执行错误(如rm了一个不存在的文件)则忽略,继续往下执行(不推荐这样做!我们可以用rm-f xxx代替-rm xxx)。

4.

makefile中可以使用“includexxx”来包含其他的makefile文件,和c语言是一样的。

5.

默认情况下,make会输出它执行的每一条shell命令,如“echohello”会输出:

echohello

hello

第一行是命令本身,第二行才是命令的执行结果

我们可以在命令前加“@”禁止输出命令,只输出执行结果

6.

Makefile中对“$”表示对变量或者函数的引用,如果我们的规则需要“$”,需要书写两个连续的(“$$”)。如果需要统配符最为字符,可以这样:\*\?

7.

默认的情况下,make会在工作目录(执行make的目录)下按照文件名顺序寻找文件名的顺序:

GNUmakefile-->makefile-->Makefilemakefile文件,推荐使用“makefile”和“Makefile”

其中更推荐“Makefile ”,因为首字母大写而比较显著,一般在一个目录中和当前目录的一些重要文件(READMEChagelist等)靠近,在寻找时会比较容易的发现它。

我们也可以自己指定makefile文件名,需要通过make的“-f xxx”或者“--file xxx”选项。也可以通过多个“-f”或者“--file”选项来指定多个需要读取的makefile文件,多个makefile文件将会被按照指定的顺序进行连接并被make解析执行。当通过 “-f”“--file”指定make读取makefile的文件时,make就不再自动查找这三个标准命名的makefile文件。

8.

定义变量时如果:object= *.o,那么$(object)只表示“*.o”而不是所有以.o结尾的文件!如果需要变量“objects”代表当前目录下的所有的.o文件,则需要是用 函数“wildcard”来实现:objects= $(wildcard *.o)





make常用命令行参数:

-w

在进入一个目录时输出:

make:Entering directory xxx.

在离开一个目录时输出:

make:Leaving directory xxx.






1.定义变量:

Makefile中的变量大小写敏感,Makefile传统做法是变量名是全采用大写的方式。

推荐的做法是在对于内部定义定义的一般变量(例如:目标文件列表objects)使用小写方式, 而对于一些参数列表 (例如:编译选项CFLAGS)采用大写方式。


变量定义完后可以“$()”或“${}”方式引用

特殊变量:“$<”“$@”“$?”“$*”



=”,“:=”,“”在定义变量时的区别:

=”表示如果一个变量引用了另一个变量,那么它在定义时不会展开,在引用时才展开。它的优点是可以引用在当前行之后定义的变量。

:=”表示如果一个变量引用了另一个变量,那么它在定义时就直接展开,得到一个不包含任何其他引用的字符串。缺点是不能引用之后定义的变量。如果引用一个未定义的变量,则为空

“?=”表示如果此变量在之前没有定义,就赋值,否则保持原值

例:

a = $(b)
b = good
target:
@echo$(a)

和:

a := $(b)
b = good
target:
@echo$(a)

第一个输出”good“,第二个输出”“


追加变量:

通常对于一个通用变量在定义之后的其他一个地方,需要给它的值进行追加。在Makefile中使用“+=”(追加方式)来实现对一个变量值的追加操作。像下边那样:

objects+= another.o

有些需要注意的地方:

1.

如果被追加值的变量之前没有定义,那么,“+=”会自动变成“=”,此变量就被定义为一 个递归展开式的变量。

2.

如果之前存在这个变量定义,那么“+=”就继承之前定义时的变量风格(可参考5.2两种变量定义 一节)

variable := value
variable += more

等价与:

variable := value
variable := $(variable) more


variable = value
variable += more

等价与:

temp = value
variable = $(temp) more


我们可以用define来定义一个多行变量:

define”定义变量的语法格式:以指示符“define”开始,“endif”结束,之间的所有内容就是所定义变量的值。所要定义的变量名字指示符“define”的同一行之后;指示符所在行的下一行开始一直到“end”所在行的上一行之间的若干行,是变量的值定义。

definetwo-lines
echofoo
echo$(bar)
endef

注意,define定义等价与“=”定义,只不过定义了多行。

上面的例子等价与:

two-lines= echo foo; echo $(bar)




Makefile中可以使用所有的环境变量,它们作为默认已定义的变量,$(xxx)便可引用,如echo$(PATH)



如果在上层makefile中定义了一个变量,那么它只在这个makefile中有效,如果要让它在子makefile中仍然有效,需要:exportxxx




Makefile条件语句:

条件语句的一般格式为:

ifeq …
  xxx
endif

ifeq …
  xxx
else …
  yyy
endif



合法的ifeq格式:

ifeq (ARG1, ARG2)

ifeq ‘ARG1’ ‘ARG2‘

ifeq ”ARG1” “ARG2“


合法的ifneq格式:

ifneq (ARG1, ARG2)

ifneq ‘ARG1’ ‘ARG2‘

ifneq ”ARG1” “ARG2“


ifdef用来判断一个变量是否定义:ifdef xxx

注意:只有在xxx没有定义或者这样定义:xxx=ifdef才是false,其他均为true,如:

bar =
foo =$(bar)
ifdef foo
  @echoyes
else
  @echono
endif

输出:yes,虽然展开后foo=


ifndef用来判断一个变量是否没有定义:ifndef xxx,和ifdef正好相反




Makefile函数:

Makefile中引用函数:$(函数名参数)

注意:上面的函数名应该是make内嵌的函数名。对于用户自己的函数需要通过make的“call”函数来间接调用;参数之间用逗号分割

常用函数简介:

$(substFROM,TO,TEXT)

函数名称:字符串替换函数—subst

函数功能:把字串“TEXT”中的“FROM”字符替换为“TO”

返回值:替换后的新字符串。

示例:

$(substee,EE,feet on the street)

替换“feeton the street”中的“ee”为“EE”

,结果是新的字符串“fEEton the strEEt”


GNU MAKE的详细中文手册,目录如下: 目 录 第一章:概述 1.1 概述 1.2 准备知识 第二章:GNU make 介绍 2.1 Makefile简介 2.2 Makefile规则介绍 2.3 简单的示例 2.4 make如何工作 2.5 指定变量 2.6 自动推导规则 2.7 另类风格的makefile 2.8 清除工作目录过程文件 第三章:Makefile 总述 3.1 Makefile的内容 3.2 makefile文件的命名 3.3 包含其它makefile文件 3.4 变量 MAKEFILES 3.5 变量 MAKEFILE_LIST 3.6 其他特殊变量 3.7 makefile文件的重建 3.8 重载另外一个makefile 3.9 make如何解析makefile文件 3.9.1 变量取值 3.9.2 条件语句 3.9.3 规则的定义 3.10 总结 第四章:Makefile的规则 4.1 一个例子 4.2 规则语法 4.3 依赖的类型 4.4 文件名使用通配符 4.4.1 统配符使用举例 4.4.2 通配符存在的缺陷 4.4.3 函数wildcard 4.5 目录搜寻 4.5.1 一般搜索(变量VPATH) 4.5.2 选择性搜索(关键字vpath) 4.5.3 目录搜索的机制 4.5.4 命令行和搜索目录 4.5.5 隐含规则和搜索目录 4.5.6 库文件和搜索目录 4.6 Makefile伪目标 4.7 强制目标(没有命令或依赖的规则) 4.8 空目标文件 4.9 Makefile的特殊目标 4.10 多目标 4.11 多规则目标 4.12 静态模式 4.12.1 静态模式规则的语法 4.12.2 静态模式和隐含规则 4.13 双冒号规则 4.14 自动产生依赖 第五章:规则的命令 5.1 命令回显 5.2 命令的执行 5.3 并发执行命令 5.4 命令执行的错误 5.5 中断make的执行 5.6 make的递归执行 5.6.1 变量MAKE 5.6.2 变量和递归 5.6.3 命令行选项和递归 5.6.4 -w选项 5.7 定义命令包 5.8 空命令 第六章:Makefile中的变量 6.1 变量的引用 6.2 两种变量定义(赋值 ) 6.2.1 递归展开式变量 6.2.2 直接展开式变量 6.2.3 如何定义一个空格 6.2.4 “?=”操作符 6.3 变量的高级用法 6.3.1 变量的替换引用 6.3.2 变量的套嵌引用 6.4 变量取值 6.5 如何设置变量 6.6 追加变量值 6.7 override 指示符 6.8 多行定义 6.9 系统环境变量 6.10 目标指定变量 6.11 模式指定变量 第七章:Makefile的条件执行 7.1 一个例子 7.2 条件判断的基本语法 7.2.1 关键字“ifeq” 7.2.2 关键字“ifneq” 7.2.3 关键字“ifdef” 7.2.4 关键字“ifndef” 7.3 标记测试的条件语句 第八章:make的内嵌函数 8.1 函数的调用语法 8.2 文本处理函数 8.2.1 $(subst FROM,TO,TEXT) 8.2.2 $(patsubst PATTERN,REPLACEMENT,TEXT) 8.2.3 $(strip STRINT) 8.2.4 $(findstring FIND,IN) 8.2.5 $(filter PATTERN…,TEXT) 8.2.6 $(filter-out PATTERN...,TEXT) 8.2.7 $(sort LIST) 8.2.8 $(word N,TEXT) 8.2.9 $(wordlist S,E,TEXT) 8.2.10 $(words TEXT) 8.2.11 $(firstword NAMES…) 8.3 文件名处理函数 8.3.1 $(dir NAMES…) 8.3.2 $(notdir NAMES…) 8.3.3 $(suffix NAMES…) 8.3.4 $(basename NAMES…) 8.3.5 $(addsuffix SUFFIX,NAMES…) 8.3.6 $(addprefix PREFIX,NAMES…) 8.3.7 $(join LIST1,LIST2) 8.3.8 $(wildcard PATTERN) 8.4 foreach 函数 8.5 if 函数 8.6 call函数 8.7 value函数 8.8 eval函数 8.9 origin函数 8.10 shell函数 8.11 make的控制函数 8.11.1 $(error TEXT…) 8.11.2 $(warning TEXT…) 第九章:执行make 9.1 指定makefile文件 9.2 指定终极目标 9.3 替代命令的执行 9.4 防止特定文件重建 9.5 替换变量定义 9.6 使用make进行编译测试 9.7 make的命令行选项 第十章:make的隐含规则 10.1 隐含规则的使用 10.2 make的隐含规则一览 10.3 隐含变量 10.3.1 代表命令的变量 10.3.2 命令参数的变量 10.4 make隐含规则链 10.5 模式规则 10.5.1 模式规则介绍 10.5.2 模式规则示例 10.5.3 自动化变量 10.5.4 模式的匹配 10.5.5 万用规则 10.5.6 重建内嵌隐含规则 10.6 缺省规则 10.7 后缀规则 10.8 隐含规则搜索算法 第十一章:使用make更新静态库文件 11.1 库成员作为目标 11.2 静态库的更新 11.2.1 更新静态库的符号索引表 11.3 make静态库的注意事项 11.4 静态库的后缀规则 第十二章 : GNU make的特点 12.1 源自System v的特点 12.2 源自其他版本的特点 12.3 GNU make自身的特点 第十三章 和其它版本的兼容 第十四章 Makefile的约定 14.1 基本的约定 14.2 规则命令行的约定 14.3 代表命令变量 14.4 安装目录变量 14.5 Makefile的标准目标名 14.6 安装命令分类 第十五章 make的常见错误信息   附录:关键字索引 1. GNU make可识别的指示 符 2. GNU make函数 3. GNU make的自动化变量 4. GNU make环境变量 后序
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值