学会使用makefile:4

这里主要介绍一下如何处理编译文件以来的问题:

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
    gcc $^ -o $@
main.o: main.h stack.h maze.h
stack.o: stack.h main.h
maze.o: maze.h main.h
clean:
    -rm main *.o
.PHONY: clean

一般一概将makefile写成下面这种形式:

all: main
main: main.o stack.o maze.o
    gcc $^ -o $@
clean:
    -rm main *.o.PHONY: clean
sources = main.c stack.c maze.c
include $(sources:.c=.d)
%.d: %.c
    set -e; rm -f $@; \
    $(CC) -MM $(CPPFLAGS) $< > $@.$$$$; \
    sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \
    rm -f $@.$$$$


source这一项指出了所有的要编译的.c文件, $(sources: .c = .d),相当于讲所有的.c文件都替换成.d文件。

那么include那一项实际上就相当于 include main.d stack.d maze.d

但是当前目录下实际上目前是没有.d文件的,但是 make 会把 include 的文件名也当作目标来尝试更新,而这些目标适用模式规则 %.d: %c ,所以执行它的命令列表,比如生成 maze.d 的命令:

set -e; rm -f maze.d; \
    cc -MM maze.c > maze.d.$$; \
    sed 's,\(maze\)\.o[ :]*,\1.o maze.d : ,g' < maze.d.$$ >
maze.d; \
    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" ,那么:
1、 main.c 的修改日期变了,根据规则 main.o main.d: main.c main.h stack.h maze.h 要重新生成 main.o 和 main.d 。生成 main.o 的规则有两条:

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.o main.d: main.c main.h stack.h maze.h 拆开写得到的,第二条是隐含规则,因此执行 cc 命令重新编译 main.o 。生成 main.d 的规则也有两条:

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 $@.$$$$


因此 main.d 的内容被更新为 main.o main.d: main.c main.h stack.h maze.h foo.h 。
2、由于 main.d 被Makefile包含, main.d 被更新又导致 make 重新读取整个Makefile,把新
的 main.d 包含进来,于是新的依赖关系生效了。

 

 

转载于:https://my.oschina.net/u/2310741/blog/729136

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值