前面我们讲了使用隐式规则可以让make在必要时自动创建.o文件的规则,但make的隐式规则的命令是固定的,对于xyz.o: xyz.c,它实际上是:

  1. $(CC) $(CFLAGS) -c -o $@ $<

能修改的只有变量$(CC)$(CFLAGS)。如果要执行多条命令,使用隐式规则就不行了。

这时,我们可以自定义模式规则(Pattern Rules),它允许make匹配模式规则,如果匹配上了,就自动创建一条模式规则。

我们修改上一节的Makefile如下:

  1. OBJS = $(patsubst %.c,%.o,$(wildcard *.c))
  2. TARGET = world.out
  3. $(TARGET): $(OBJS)
  4. cc -o $(TARGET) $(OBJS)
  5. # 模式匹配规则:当make需要目标 xyz.o 时,自动生成一条 xyz.o: xyz.c 规则:
  6. %.o: %.c
  7. @echo 'compiling $<...'
  8. cc -c -o $@ $<
  9. clean:
  10. rm -f *.o $(TARGET)

make执行world.out: hello.o main.o时,发现没有hello.o文件,于是需要查找以hello.o为目标的规则,结果匹配到模式规则%.o: %.c,于是make自动根据模式规则为我们动态创建了如下规则:

  1. hello.o: hello.c
  2. @echo 'compiling $<...'
  3. cc -c -o $@ $<

查找main.o也是类似的匹配过程,于是我们执行make,就可以用模式规则完成编译:

  1. $ make
  2. compiling hello.c...
  3. cc -c -o hello.o hello.c
  4. compiling main.c...
  5. cc -c -o main.o main.c
  6. cc -o world.out hello.o main.o

模式规则的命令完全由我们自己定义,因此,它比隐式规则更灵活。

但是,模式规则仍然没有解决修改hello.h头文件不会触发main.c重新编译的问题,这个依赖问题我们继续放到后面解决。

最后注意,模式规则是按需生成,如果我们在当前目录创建一个zzz.o文件,因为make并不会在执行过程中用到它,所以并不会自动生成zzz.o: zzz.c这个规则。

参考源码

可以从GitHub下载源码。

GitHub

小结

使用模式规则可以灵活地按需动态创建规则,它比隐式规则更灵活。

查看官方手册: