参考文件是: 跟我一起写 Makefile 作者:陈皓
一、静态模式
语法格式:
例子:
上面的例子中,指明了我们的目标从$object中获取,“%.o”表明要所有以“.o”结尾的目标,也就是“foo.o bar.o”,也就是变量$object集合的模式,而依赖模式“%.c”则取模式“%.o”的“%”,也就是“foo bar”,并为其加下“.c”的后缀,于是,我们的依赖目标就是“foo.c bar.c”。而命令中的“$<”和“$@”则是自动化变量,“$<”表示所有的依赖目标集(也就是“foo.c bar.c”),“$@”表示目标集(也就是“foo.o bar.o”)。
静态规则的另一个例子:
第一行就是先定义一个变量files,其变量的内容包括:foo.elc bar.o lose.o,第二行中的filter函数是用来过滤的一个功能函数,即将files中以.o结尾的文件过滤出来,后面的静态模式,就是将以与.o文件同名的.c文件作为指令的依赖,$(CC)这个代表CC这个命令,这个是经过重新定义的,一般指gcc也有可能是其他的。$<这个表示其依赖文件是一类,这里指的是前面说的与.o文件同名的.c文件,$@这个表示其目标,即最终的.o文件,后面两行意思差不多。
二、 自动生成依赖性
自动生成依赖性的意思就是自动将依赖文件中的.c文件相关的.h文件匹配好,如果你使用GNU的C/C++编译器,你得用“-MM”参数,不然,“-M”参数会把一些标准库的头文件也包含进来。
上述语句中的“$(sources:.c=.d)”中的“.c=.d”的意思是做一个替换,把变量$(sources)所有[.c]的字串都替换成[.d]。
命令的显示
通常,make会把其要执行的命令行在命令执行前输出到屏幕上。当我们用“@”字符在命令行前,那么,这个命令将不被make显示出来,最具代表性的例子是,我们用这个功能来像屏幕显示一些信息。如:
@echo 正在编译XXX模块......
当make执行时,会输出“正在编译XXX模块......”字串,但不会输出命令,如果没有“@”,那么,make将输出:
echo 正在编译XXX模块......
正在编译XXX模块......
如果make执行时,带入make参数“-n”或“--just-print”,那么其只是显示命令,但不会执行命令,这个功能很有利于我们调试我们的Makefile,看看我们书写的命令是执行起来是什么样子的或是什么顺序的。
而make参数“-s”或“--slient”则是全面禁止命令的显示。
变量传递
如果你要传递变量到下级Makefile中,那么你可以使用这样的声明:
export <variable ...>
如果你不想让某些变量传递到下级Makefile中,那么你可以这样声明:
unexport<variable ...>
如果你要传递所有的变量,那么,只要一个export就行了。后面什么也不用跟,表示传递所有的变量。
有两个变量,一个是SHELL,一个是MAKEFLAGS,这两个变量不管你是否export,其总是要传递到下层Makefile中,特别是MAKEFILES变量,其中包含了make的参数信息,如果我们执行“总控Makefile”时有make参数或是在上层Makefile中定义了这个变量,那么MAKEFILES变量将会是这些参数,并会传递到下层Makefile中,这是一个系统级的环境变量。
但是make命令中的有几个参数并不往下传递,它们是“-C”,“-f”,“-h”“-o”和“-W”
“-w”或是“--print-directory”会在make的过程中输出一些信息,让你看到目前的工作目录,其会在进入目录前打印出:make: Entering directory`/home/hchen/gnu/make'.
而在完成下层make后离开目录时,我们会看到: make: Leaving directory `/home/hchen/gnu/make'
subst函数是替换的功能