20180417 makefile学习笔记-书写规则(静态模式)
静态模式可以更加容易地定义多目标的规则,可以让我们的规则变得更加有弹性和灵活性,静态模式的语法是:
<targets...>:<target-pattern>:<prereq-patterns...>
<commands>
...
targets定义了一系列的目标文件,可以有通配符,是目标的一个集合;
target-pattern是指明了targets的模式,也就是目标集模式;
prereq-patterns是目标的依赖模式,它对target-pattern形成的模式再进行一次依赖目标的定义。
比如如果我们的<target-pattern>定义成“%.o”,意思是我们的<targets...>集合中都是以.o结尾的,而如果我们的<prereq-patterns...>定义成“%.c”,意思是对<target-pattern>所形成的目标集进行二次定义,其计算方法是,取<target-pattern>模式里的“%”(也就是去掉[.o]这个结尾),并为其加上[.c]这个结尾,形成的新集合。
所以,我们的“目标模式”或是“依赖模式”中都应该有“%”这个字符,如果你的文件名中有“%”,那么你可以使用反斜杠“\”进行转义,来标明真实的“%”字符。eg:
objects=foo.o bar.o
all:$(objects)
$(objects):%.o:%.c
$(CC) -c $(CFLAGS) $< -o $@
上面的例子中,指明了我们的目标从$objects中获取,“%.o”标明要所有以“%.o”结尾的目标,也就是“foo.o bar.o”,也就是变量$objects集合的模式,而依赖模式“%.c”则取模式“%.o”的“%”,也就是“foo bar”,并为其加上“%.c”后缀,于是,我们的依赖目标就是“foo.c bar.c”,而命令中的“$<”“$@”则是自动化变量,“$<”表示所有的依赖目标集(也就是)“foo.c bar.c”,“$@”表示目标集(也就是“foo.o bar.o”)。于是,上面的规则展开后就等价于下面的规则:
foo.o:foo.c
$(CC) -c $(CFLAGS) foo.c -o foo.o
bar.o:bar.c
$(CC) -c $(CFLAGS) bar.c -o bar.o
试想,如果我们的“%.o”有几百个,那样我们只要用这种很简单的“静态模式规则”就可以写完一堆规则,效率大大的提高了,“静态模式规则”的用法很灵活,如果用的好,那会是很强大的功能。再看一个例子:
files=foo.elc bar.o lose.o
$(filter %.o,$(files)):%.o:%.c
$(CC) -c $(CFLAGS) $< -o $@
$(filter %.elc,$(files)):%.elc:%.el
emacs -f batch-byte-compile $<
$(filter %.o,$(files))表示调用Makefile的filter函数,过滤“$filter”集,只要其中模式为“%.o”的内容。其他内容类似。这个例子展示了Makefile中更大的弹性。