makefile的语法规则
目标(target):依赖文件1 依赖文件2 ......
<tab> command
- 在makefile当中的#代表批注;
- <tab>需要在命令行的第一个字符(例如上面gcc这个编译程序命令);
- 目标(target)与相关文件(就是目标文件)之间需要以“:”隔开。
Linux操作系统真正认识的是二进制程序,当我们编辑完程序后实际上是一个纯文本文件。所以我们需要对该纯文本文件进行编译——即生成二进制文件,编译之后进行链接就会将二进制文件生成可执行的二进制文件,此时操作系统就可以执行这个二进制程序了。
示例
一个工程里有3个源文件和2个头文件,makefile文件如下:
edit:server.o head.o find.o
gcc -o edit server.o head.o find.o -levent
server.o:server.c head.c find.c
gcc -c server.c head.c find.c
head.o:head.c find.c
gcc -c head.c find.c
find.o:find.c
gcc -c find.c
clean:
rm -fr edit server.o head.o find.o
- edit表示目标文件,server.o head.o find.o表示目标文件的依赖文件;
- gcc表示make执行的命令即生成可执行目标文件edit;
- .o文件“:”之后的文件表示.o文件的依赖文件,下面的gcc表示生成该.o文件
- -levent表示生成可执行文件时需要依赖的库文件event,-l后面跟依赖的库文
- clean表示make后删除可执行文件以及生成的.o文件
makefile使用变量
使用变量来写makefile使得makefile更加简洁,但依赖关系就没有那么明显了
示例
objects=server.o head.o find.o
edit:$(objects)
gcc -o edit $(objects) -levent
server.o:head.h find.h
head.o:head.h find.h
find.o:find.h
clean:
rm edit $(objects)
make的工作方式
- 读入所有的makefile;
- 读入被include的其他makefile;
- 初始化变量;
- 推到隐晦规则(例如上例中的.o文件与.c文件的关系)
- 为所有的目标文件创建依赖关系链;
- 根据依赖关系决定哪些目标文件重新生成;
- 执行生成命令;
makefile的自动化变量
$@
表示规则中的目标文件集。在模式规则中,如果有多个目标,那么,"$@"就是匹配于目标中模式定义的集合。
$%
仅当目标是函数库文件中,表示规则中的目标成员名。例如,如果一个目标是"foo.a(bar.o)",那么,"$%"就是"bar.o","$@"就是"foo.a"。如果目标不是函数库文件(Unix下是[.a],Windows下是[.lib]),那么,其值为空。
$
依赖目标中的第一个目标名字。如果依赖目标是以模式(即"%")定义的,那么"$将是符合模式的一系列的文件集。注意,其是一个一个取出来的。
$?
所有比目标新的依赖目标的集合,以空格分隔。
$^
所有的依赖目标的集合,以空格分隔。如果在依赖目标中有多个重复的,那个这个变量会去除重复的依赖目标,只保留一份。
$+
这个变量很像"$^",也是所有依赖目标的集合。只是它不去除重复的依赖目标。
$*
这个变量表示目标模式中"%"及其之前的部分。如果目标是"dir/a.foo.b",并且目标的模式是"a.%.b",那么,"$*"的值就是"dir/a.foo"。这个变量对于构造有关联的文件名是比较有效。如果目标中没有模式的定义,那么"$*"也就不能被推导出,但是,如果目标文件的后缀是make所识别的,那么"$*"就是除了后缀的那一部分。例如:如果目标是"foo.c",因为".c"是make所能识别的后缀名,所以,"$*"的值就是"foo"。这个特性是GNU make的,很有可能不兼容于其它版本的make,所以,你应该尽量避免使用"$*",除非是在隐含规则或是静态模式中。如果目标中的后缀是make所不能识别的,那么"$*"就是空值。