Makefile的基本形式如下所示:
TARGET : DEPENDENCES
CMD
# TARGET:生成的目标,可以是一个文件,也可以是一个虚拟符号(非真实文件)
# DEPENDENCES: 生成目标的所有依赖,它是一个集合,可以只有一个文件,或者很多文件;也可以是虚拟符号
# CMD:把所有依赖生成目标的执行命令,其中CMD前面是一个TAB键,而不能是空格
Makefile的基本规则,归纳起来就是一句话:【当一个TARGET (欲生成的目标)比它的任何一个DEPENDENCES(依赖的文件)旧时,这个TARGET就要重新生成】。
这句话读起来很简单,就是Makefile在执行编译的时候,会先去判断TARGET和其依赖文件的时间,如果TARGET的时间比较旧,意味着依赖文件有更新了,所以TARGET要重新生成;反之,如果TARGET的时间比较新,意味着在TARGET生成后,依赖文件是没有改变过的,自然就不用重新去生成TARGET
FORCE 在Makefile的含义
有了上一小节中介绍的Makefile基本概念后,我们来进一步分析下Makefile中的FORCE,以下是FORCE在Makefile中出现的最简化版本:
file = test.txt
all: generate-a-file
generate-a-file: $(file)
$(file):
@echo "Force to generate a test file for every make ..."
rm -rf $(file) && echo `date "+%Y-%m-%d %H:%M:%S"` > $(file)
FORCE:
.PHONY: FORCE
这个Makefile不是用于真正的编译工程,而是提供一个很简单的功能,生成一个test.txt,并且这个test.txt的内容是记录每次编译的时间。但是使用这个makefile执行make时,发现只有第一次make的时候,才会生成test.txt,而其他时候只要test.txt还存在都不会重新生成,如下所示:
很明显,它没有达到我们期望的“每次编译都重新生成test.txt”。这个时候 FORCE 就发挥作用了,我们在 test.txt目标的后面添加 FORCE作为它的依赖试试看,即如下所示:
file = test.txt
all: generate-a-file
generate-a-file: $(file)
$(file): FORCE #FORCE表示每次这段都要执行
@echo "Force to generate a test file for every make ..."
rm -rf $(file) && echo `date "+%Y-%m-%d %H:%M:%S"` > $(file)
FORCE:
.PHONY: FORCE
执行输出如下所示: 我们可以看到,这达到了我们的目的,每次test.txt都是重新生成了,它记录了每次make的时间。