makefile中的目标的含义:
默认情况下,
- make认为目标对应着一个文件
- make比较目标文件和依赖文件的新旧关系,决定是否执行命令
- make以文件处理作为第一优先级
编程实验1:(make默认认为目标对应着一个文件)
当前目录结构如下:
func.c源文件如下:void fun(){ printf("void fun():hello makefile \n");}复制代码
main.c源文件如下:
extern void fun();int main(){ fun(); return 0;}复制代码
makefile源文件如下:
hello.out:main.o func.o gcc -o hello.out main.o func.omain.o:main.c gcc -o main.o -c main.c func.o:func.c gcc -o func.o -c func.c clean: rm *.o hello.out复制代码
命令行执行结果如下:
分析: 执行make clean命令会将当前目录下的所有.o文件和hello.out一起删除;而当再次执行make clean的时候由于当前目录下已经没有.o文件和hello.out文件,所以会出现提示信息无法移除文件编程实验2:(make以文件处理作为第一优先级)
当前目录结构如下:
main.c func.c makefile文件内容均与编程实验1保持一致,命令行执行结果如下:分析: 由于当前目录下存在文件clean,故当执行make clean的命令时,make以文件处理作为第一优先级,而当前文件已经是最新的,故不会再执行make clean命令。
出现此问题的解决方法为增加伪目标,,具体分析请看编程实验3。编程实验3:(伪目标的引入,保证与目标的依赖无关,目标中的命令总是被执行)
当前目录结构如下:
main.c func.c 文件内容均与编程实验1保持一致,makefile文件做如下修改(在clean目标之前增加代码:.PHONY:clean):## makefilehello.out :main.o func.o gcc -o hello.out main.o func.omain.o:main.c gcc -o main.o -c main.cfunc.o:func.c gcc -o func.o -c func.c.PHONY:cleanclean: rm *.o hello.out复制代码
命令行执行结果如下:
伪目标知识:
- 通过.PHONY关键字声明一个伪目标
- 伪目标不对应任何实际的文件
- 不管伪目标的依赖是否更新,命令总是执行
- 伪目标的语法:先声明,后使用
本质:
伪目标是make中特殊目标.PHONY的依赖
编程实验4(使用伪目标模拟C语言中的函数调用)
当前目录结构如下:
main.c func.c 文件内容均与编程实验1保持一致,makefile文件做如下修改(在clean目标之前增加代码:.PHONY:clean):## makefilehello.out :main.o func.o gcc -o hello.out main.o func.omain.o:main.c gcc -o main.o -c main.cfunc.o:func.c gcc -o func.o -c func.c.PHONY:rebuild clean allrebuild:clean allall:hello.outclean: rm *.o hello.out复制代码
- 原理: 当一个目标的依赖包含伪目标时,伪目标所定义的命令总是会被执行 命令行执行结果如下:
编程实验5(绕开.PHONY关键字定义伪目标 )
- 原理: 如果一个规则没有命令或者依赖,并且它的目标不是一个存在的文件名;在执行此规则时,目标总是会被认为是最新的 当前目录结构如下: main.c func.c 文件内容均与编程实验1保持一致,makefile文件做如下修改(将clean目标依赖于FORCE,而FORCE定义为空目标):
##makefilehello.out :main.o func.o gcc -o hello.out main.o func.omain.o:main.c gcc -o main.o -c main.cfunc.o:func.c gcc -o func.o -c func.cclean:FORCE rm *.o hello.outFORCE:复制代码
命令行执行结果如下:
- 分析: 因为FORCE目标是最新的,所以每次执行make clean的时候都会执行clean的命令
小结:
- 默认情况下,make认为目标对应着一个文件
- .PHONY用于声明一个伪目标,伪目标不对应实际的文件
- 伪目标的本质是make中特殊目标.PHONY的依赖
- 使用伪目标可以模拟“函数调用”,方法请参考上文编程实验5