这里主要介绍一下如何处理编译文件以来的问题:
main.o: main.h stack.h maze.h stack.o: stack.h main.h maze.o: maze.h main.h可见上面几个文件都依赖与几个头文件,实际上自己一个一个动手去写会比较麻烦,而且还容易弄错。可以用 gcc 的 -M 选项自动生成目标文件和源文件的依赖关系。如果像去除对系统库文件的以来关系,可以去使用gcc -MM
举个例子,来说明吧:
一个普通的makefile如下所示:
all: main main: main.o stack.o maze.o <span style="white-space:pre"> </span>gcc $^ -o $@ main.o: main.h stack.h maze.h stack.o: stack.h main.h maze.o: maze.h main.h clean: <span style="white-space:pre"> </span>-rm main *.o .PHONY: clean
一般一概将makefile写成下面这种形式:
all: main main: main.o stack.o maze.o <span style="white-space:pre"> </span>gcc $^ -o $@ clean: <span style="white-space:pre"> </span>-rm main *.o.PHONY: clean sources = main.c stack.c maze.c include $(sources:.c=.d) %.d: %.c <span style="white-space:pre"> </span>set -e; rm -f $@; \ <span style="white-space:pre"> </span>$(CC) -MM $(CPPFLAGS) $< > $@.$$$$; \ <span style="white-space:pre"> </span>sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ <span style="white-space:pre"> </span>rm -f $@.$$$$
那么include那一项实际上就相当于 include main.d stack.d maze.d
但是当前目录下实际上目前是没有.d文件的,但是 make 会把 include 的文件名也当作目标来尝试更新,而这些目标适用模式规则 %.d: %c ,所以执行它的命令列表,比如生成 maze.d 的命令:
set -e; rm -f maze.d; \ <span style="white-space:pre"> </span>cc -MM maze.c > maze.d.$$; \ <span style="white-space:pre"> </span>sed 's,\(maze\)\.o[ :]*,\1.o maze.d : ,g' < maze.d.$$ > maze.d; \ <span style="white-space:pre"> </span>rm -f maze.d.$$上面这一段的执行步骤实际上如下所示:
1.set -e;命令设置当前Shell进程为这样的状态:如果它执行的任何一条命令的退出状态非零则立刻终止,不再执行后续命令。
2.删除原来的maze.d
3. 通过gcc的-MM选项重新生成maze.c的依赖关系,两个$$的主要作用是表示当前进程的id,一般用于生成唯一的文件名。
4.sed进行逐行的处理,maze.d.1234 的内容应该是 maze.o: maze.c maze.h main.h ,经过 sed 处理之后存为 maze.d ,其内容是 maze.o maze.d: maze.c maze.h main.h 。
5. 最后删除maze.d.1234文件。
不管是Makefile本身还是被它包含的文件,只要有一个文件在 make 过程中被更新了, make 就会重新读取整个Makefile以及被它包含的所有文件,现在 main.d 、 stack.d 和 maze.d 都生成了,就可以正常包含进来了(假如这时还没有生成, make 就要报错而不是报警告了),相当于在Makefile中添了三条规则:
main.o main.d: main.c main.h stack.h maze.h maze.o maze.d: maze.c maze.h main.h stack.o stack.d: stack.c stack.h main.h如果我在 main.c 中加了一行 #include "foo.h" ,那么:
main.o: main.c main.h stack.h maze.h %.o: %.c # commands to execute (built-in): $(COMPILE.c) $(OUTPUT_OPTION) $<第一条是把规则 main.o main.d: main.c main.h stack.h maze.h 拆开写得到的,第二条是隐含规则,因此执行 cc 命令重新编译 main.o 。生成 main.d 的规则也有两条:
main.o: main.c main.h stack.h maze.h %.o: %.c # commands to execute (built-in): $(COMPILE.c) $(OUTPUT_OPTION) $<
main.d: main.c main.h stack.h maze.h %.d: %.c set -e; rm -f $@; \ $(CC) -MM $(CPPFLAGS) $< > $@.$$$$; \ sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ rm -f $@.$$$$