Makefile编写的时候会需要定义一些变量,这时候就需要用到“=”也就是C语言中的赋值符号来进行赋值,但有时候也会遇到":="和"?=",那么它们的区别是什么呢?
举个例子:
编写一个Makefile
A = "equal"
all:
@echo $(A)
执行make命令,输出equal。这是普通的变量赋值,用了“=”。有点类似于宏的定义,但与宏定义又有一些区别,修改代码
A = $B "equal"
B = "nocolon"
all:
@echo $(A)
执行make,输出nocolon equal,所以在使用“=”进行变量赋值的时候,如果右边有变量,变量在本句之后定义的,依旧不会报错,会用之后定义的变量来替换。也就是用“=”定义变量的时候,右边如果有变量,则这个变量的定义可以是整个Makefile文件中的任意位置。
修改上述代码
B = "colon"
A := $B "equal"
all:
@echo $(A)
执行make,输出colon equal,这样看上去是和"="没有区别,但如果调换一下A、B两变量定义的位置呢?
A := $B "equal"
B = "colon"
all:
@echo $(A)
执行make,输出equal,没有colon,这样就能看出来区别了,使用“:="进行变量的赋值时,如果右边有变量,那么只会使用在这条语句之前定义的变量,如果变量此时未定义,就默认这个变量为空。
所以使用"="和":="的时候,如果右边有变量,则两者有区别,没有变量,两者是没有区别的。那么"?="呢?
B = "question"
A ?= $B "equal"
all:
@echo $(A)
执行make,输出question equal,修改代码
A ?= $B "equal"
B = "question"
all:
@echo $(A)
执行make,依旧输出question equal,说明在正常赋值这方面来说,"?="是和"="一样使用的。那它们的不同之处在哪里呢?修改代码
A ="equal"
A ?= "question equal"
all:
@echo $(A)
执行make,输出结果为equal。说明了使用"?="进行赋值的时候如果该变量已经赋值过了,那么将跳过这一句,使用"="的时候,如果变量已经赋值过了,它会在进行一次赋值。
Makefile编写的时候还有一个+=符号,+=符号又有什么用处呢?
A = "plus"
A += "equal"
all:
@echo $(A)
执行make,输出结果为plus equal,所以+=的意思和编程语言中的+=用法差不多,是在本变量的后面加上一个字符串。
所以":="和"?="也都是赋值,但都是对"="进行了一定的限制,":="是在右边有变量的时候只是用本句之前定义的变量,"?="是先判断该变量是否定义过,定义过就不覆盖定义了。