makefile进阶(1)——两种不同的依赖类型:normal prerequisites和order-only prerequisites

一、normal prerequisites和order-only prerequisites的区别

通常,如果规则中依赖文件中的任何一个被更新,则规则的目标相应地也应该被更新。有时,需要定义一个这样的规则,在更新目标(目标文件已经存在)时只需要根据依赖文件中的部分来决定目标是否需要被重建,而不是在依赖文件的任何一个被修改后都重建目标。为了实现这一目的,相应的就需要对规则的依赖进行分类,一类是在这些依赖文件被更新后,需要更新规则的目标;另一类是更新这些依赖的,可不需要更新规则的目标。我们把第二类称为:“order-only”依赖。书写规则时,“order-only”依赖使用管道符号“|”开始,作为目标的一个依赖文件。规则依赖列表中管道符号“|”左边的是常规依赖,管道符号右边的就是“order-only”依赖。这样的规则书写格式如下:

TARGETS : NORMAL-PREREQUISITES | ORDER-ONLY-PREREQUISITES

注意:

1、竖线左边的常规依赖文件可以是空;

2、如果前提目标中同时存在正常前提目标和命令前提目标,则正常前提目标优先生成。

3、规则依赖文件列表中如果一个文件同时出现在常规列表和“order-only”列表中,那么此文件被作为常规依赖处理

二、实例分析与验证

实例1: 

LIBS = libtest.a
foo : foo.c | $(LIBS)
    $(CC) $(CFLAGS) $< -o $@ $(LIBS)


make在执行这个规则时,如果目标文件“foo”已经存在。当“foo.c”被修改以后,目标“foo”将会被重建,但是当“libtest.a”被修改以后。将不执行规则的命令来重建目标“foo”。
就是说,规则中依赖文件$(LIBS)只有在目标文件不存在的情况下,才会参与规则的执行。当目标文件存在时此依赖不会参与规则的执行过程。
 

实例2: 

在当前文件夹下存在文件main.c和makefile文件,

其中makefile文件内容如下:包含三条规则,为了方便后续分析,分别将其表示为规则1、2、3。

TEST_DIR=dirTest

all:main.o
	@echo "this is all"

main.o : main.c | ${TEST_DIR}
	@echo "gcc process"
	gcc -c main.c -o $@

${TEST_DIR}:
	@echo "creat the dirTest directory"
	mkdir -p dirTest

步骤一、执行make命令

makefile进阶(1)——两种不同的依赖类型:normal prerequisites和order-only prerequisites_第1张图片

通过输出的打印信息可以发现,三条规则的执行顺序分别是:规则3->2->1。此时,文件夹中的内容如下:

步骤二、在上述步骤的基础上修改dirTest文件夹的内容,我这里是在文件夹中添加了一个文件。然后执行make

通过打印可以发现,虽然更新了dirTest的内容,但是第二条规则依旧没有执行(因为dirTest已经存在),这就是order-only依赖的作用。

步骤三、在上述步骤的基础上,将order-only依赖转换为常规依赖,即将makefile修改成如下:

TEST_DIR=dirTest

all:main.o
	@echo "this is all"

main.o : main.c  ${TEST_DIR}
	@echo "gcc process"
	gcc -c main.c -o $@

${TEST_DIR}:
	@echo "creat the dirTest directory"
	mkdir -p dirTest

再次执行make

通过打印信息可以看出执行了规则2、1,符合常规依赖的执行规则。

 

ref:

https://www.gnu.org/software/make/manual/make.html#Prerequisite-Types

https://blog.csdn.net/npjocj/article/details/8025378

徐海兵 gnu make翻译版

你可能感兴趣的:(嵌入式linux)