1. Makefile的基础语法
1.1 通配符
假如一个目标文件所依赖的依赖文件很多,那样岂不是我们要写很多规则,这显然是不合乎常理的
这时候我们就可以使用通配符来解决这些问题了
这里有一个Makefile文件
test: a.o b.o
gcc -o test $^
%.o : %.c
gcc -c -o $@ $<
%.o:表示所用的.o文件
%.c:表示所有的.c文件
$@:表示目标,指的是冒号左边的文件名
$<:表示第1个依赖文件,指的是冒号右边的文件名
$^:表示所有依赖文件,指的冒号是右边所有的依赖文件
1.2 假想目标: .PHONY
当我们想清除文件的时候,我们在Makefile的结尾写:
clean:
rm *.o test
执行 make :生成第一个可执行文件还有很多依赖文件.o
执行 make clean : 清除所有文件,即执行: rm *.o test。
make后面可以带上目标名,也可以不带,如果不带目标名的话它就想生成第一个规则里面的第一个目标。
- 但是执行 make clean有一个问题:
假如我们现在的路径里有个“clean”文件,这时候运行make,出现以下提示
make: \`clean' is up to date.
现在我们的目录里面有名为“clean”的文件,目标文件是有的,并且没有依赖文件,没有办法判断依赖文件的时间。这种写法会导致:有同名的"clean"文件时,就没有办法执行make clean操作
解决办法:我们需要把目标定义为假象目标,用关键子PHONY
.PHONY: clean //把clean定义为假象目标。他就不会判断名为“clean”的文件是否存在,
rm *.o test
再运行make clean 就没问题了
1.3 变量
在makefile中有两种变量:
1. 即时变量
对于即使变量使用 “:=” 表示,它的值在定义的时候已经被确定了
2. 延时变量
对于延时变量使用“=”表示。它只有在使用到的时候才确定,在定义/等于时并没有确定下来。
想使用变量的时候使用“$”来引用,如果不想看到命令是,可以在命令的前面加上"@"符号,就不会显示命令本身。当我们执行make命令的时候,make这个指令本身,会把整个Makefile读进去,进行全部分析,然后解析里面的变量。常用的变量的定义如下:
:= # 即时变量
= # 延时变量
?= # 延时变量, 如果是第1次定义才起效, 如果在前面该变量已定义则忽略这句
\+= # 附加, 它是即时变量还是延时变量取决于前面的定义
?=: # 如果这个变量在前面已经被定义了,这句话就会不会起效果,
- 举例:
A := $(C)
B = $(C)
C = abc
#D = hello
D ?= world
all:
@echo A = $(A)
@echo B = $(B)
@echo D = $(D)
C += 123
执行:make
结果:
A =
B = abc 123
D = world
解析:
- A为什么等于空
A为即使变量,在定义时即确定,由于刚开始C的值为空,所以A的值也为空。- B = abc 123
B为延时变量,只有使用到时它的值才确定,当执行make时,会解析Makefile里面的所用变量,所以先解析C= abc,然后解析C += 123,此时,C = abc 123- D变量在前面没有定义,所以D的值为world,如果在前面添加D = hello,最后D的值为hello。