Makefile中变量如C\C++中宏一样执行时候自动原模原样展开在使用的地方,与C/C++不同得失,Makefile中可以改变其值。Makefile中变量可以使用在“目标”,“依赖目标”,“命令”或其他部分。

变量名可以包含字符数字下划线(可以数字开头),但是不应该含有“:”、“#”、“=”或是空字符,区分大小写。

一、变量的基础

声明时需要给予初始值,使用时加上“$”符号,最好用“()”或者“{}”把变量包括起来。而使用真是的“$”需要“$$”表示。

变量可以用在许多地方,如规则中的目标、依赖、命令以及新的变量。

二、变量中的变量

①使用“=”,左侧是变量,右侧是变量的值,右侧变量的值可以定义在文件的任何一处,也就是说右侧变量不一定是一已定义好的值,也可以使用后面定义的值。

例如:

foo = $(bar)

bar = $(ugh)

ugh = huh?

all:

  echo $(foo)

执行make all会打印出变量$(foo)的值“Huh?”。所以,变量是可以用后面的变量来定义的。

注:这样可能会产生递归定义:

A= $(B)

B=$(A)

这会让make陷入无限的变量展开过程中去,当然,make有检测这样的能力,并会报错。

如果变量中使用函数,这也会让make运行时非常慢。他会使用make的“wildcard”和“shell”发生不可预知的错误。

②“:=”,前面的变量不能使用后面的变量,只能使用已定义好的变量。

x:=foo

y:=$(x) bar

x:=later

等价于

y := foo bar

x := later

 

y := $(x) bar

x := foo

那么y的值是“bar”,而不是“foo, bar”

系统变量“MAKELEVEL”表示,如果make有一个嵌套执行的动作,则该变量会记录当前的Makefile调用层数.

(以下的我也还没懂,先标记)

nullstring :=
space := $(nullstring) # end of the line

nullstring是一个Empty变量,其中什么也没有,而我们的space的值是一个空格。因为在操作符的右边是很难描述一个空格的,这里采用的技术很管用,先用一个Empty变量来标明变量的值开始了,而后面采用“#”注释符来表示变量定义的终止,这样,我们可以定义出其值是一个空格的变量。请注意这里关于“#”的使用,注释符“#”的这种特性值得我们注意,如果我们这样定义一个变量:

dir := /foo/bar # directory to put the frobs in

dir这个变量的值是“/foo/bar”,后面还跟了4个空格,如果我们这样使用这样变量来指定别的目录——“$(dir)/file”那么就完蛋了。

③、另外的赋值法,“?=”表示如果没有被定义过,则赋值。如果定义过,则什么也不做。

FOO ?= bar

等价于:

ifeq ($(origin FOO), undefined)

FOO = bar

endif

三、变量高级用法

①变量值的替换

替换变量中的共有部分,格式“$(var:a=b)”或是“${var:a=b}”,表示吧变量var中的所有以“a”结尾的“a”替换成“b”

例如:

foo := a.o b.o c.o

bar := $(foo:.o=.c)

把“$(foo)”中的所有“.o”结尾全部替换为“.c”,所以“$(bar)”的值就是“a.c b.c c.c”。

另外一种前面记录过得静态模式,

foo := a.o b.o c.o

bar := $(foo:%.o=%.c)

②、“把变量的值在当成变量”

x = y

y = z

a:=$($(x))

例子中,$(x)=y, 所以$($(x))就是$(y),于是$(a)的值就是z

“函数”于“条件语句”一同使用的例子:

ifdef do_set

  func := sort

else

  func := strip

endif

bar := a b d g q c

foo := $($(func) $(bar))

示例中如果定义了"do_sort",foo:=$(sort a d b g q c),于是$(foo)就是“a b c d g q”。如果没有定义“do_sort”,则foo:=$(strip a d b g q c),调用strip函数。

“把变量的值当成变量”同样可以用在操作符的左边:

dir = foo

$(dir)_sources := $(wildcard $(dir)/*.c)

define $(die)_print

  lpr $($(dir)_sources)#该命令表示把文件发送给打印机

endef

四、追加变量

使用“+=”给变量追加值,如:

objects = main.o foo.o bar.o utils.o

objects += another.o

于是,$(objects)值就变成:“main.o foo.o bar.o utils.o another.o"

等价于:

objects = main.p foo.o bar.o utils.o 

objects := $(objects) another.o

但“+=”更为简洁,且如果前面没有定义过该变量。“+=”自动变成“=”,如果 有定义过,“+=”汇集成前次操作的赋值符。如果前一次的是“:=”,那么“+=”会以“:=”为其赋值符,如:

variable := value

variable += more

等价于:

variable := value

variable := $(variable) more

但是“=”情况却不一样,例如:

variable = value

variable += more

由于前次的赋值符是“=”,所以“+=”也会以“=”作为赋值,那么岂不会发生变量的递补归定义,这是很不好的,所以make会自动为我们解决这个问题,我们不必担心这个问题。(没懂)

五、override指示符

如果有变量是通常的make命令行参数设置,那么Makefile中对这个变量的赋值会被忽略。如果想在Makefile中设置这类参数的值,可使用“override”指示符。

override <variable>=<value>

override <variable>:=<value>

还可以追加

override <variable>+=<more text>

对于多行的变量定义,可以使用define指示符,而define指示符前,也同样可以使用override,如:

override define foo

bar

endef

定义变量foo bar

六、多行变量

使用define关键字设置变量的值可以有换行,有利于一系列的命令。(命令包就是利用该关键字)

define指示符后面跟变量的名字,重起一行定义变量的值,定义以关键字endef结束。工作方式和“=”操作符一样。变量的值可以包含函数、命令、文字或是其他变量。define定义的命令变量中同样要以【Tab】键开头

define two-lines

  echo foo

  echo $(bar)

endef

七、环境变量

make运行时的系统环境变量可以在make开始运行时被载入到Makefile文件中,但是如果Makefile中定义了这个变量,或是这个变量由make命令行带入,那么系统的环境变量的值将被覆盖(如果make指定了“-e”参数,那么系统环境变量将覆盖Makefile中定义的变量,-e, --environment-overrides环境变量覆盖Makefile文件)

因此,如果我们在环境变量中设置了“CFLAGS”环境变量,那么我们就可以在所有的Makefile中使用这个变量。这对于使用统一的编译参数很方便。如果Makefile中定义了CFLAGS,那么会使用Makefile中的这个变量,如果没有定义则使用系统的环境变量。

当make嵌套调用时,上层Makefile中定义的变量会以系统环境变量的方式传递到下层的Makefile中。默认情况下,只有通过命令行设置的变量会被传递。定义在文件中的变量传递需要export。

不推荐许多的变量定义在系统环境中,这样执行不用的Makefile拥有同一套系统变量,可能会有麻烦。

八、目标变量

前面所述的Makefile中定义的变量都是“全局变量”,整个文件中都可以访问这些变量。当然“自动化变量”除外("$<"、“$@”、“$^”)属于“规则型变量”,这种变量的值依赖于规则的目标和依赖目标的定义。

 当然,也可以为某个目标设置局部变量,成为“Target-specific Variable”,它可以和“全局变量”同名,因为他的作用范围只在这条规则以及连带规则中,其值也只在作用范围内有效,不会影响规则链以外的全局变量的值。如下:

<target...>:<variable-assignment>

<target...>:overide<vatiable-assignment>

<vatiable-assignment>可以是前面的各种赋值表达式。“=”、“:=”、“+=”、“?=”。第二条是make命令行带入的变量或是系统环境变量。

示例:

prog:CFLAGS=-g

prog:prog.o foo.o bar.o

  $(CC) $(CFLAGS) prog.o foo.o bar.o

prog.o:prog.c

  $(CC) $(CFLAGS) prog.c

foo.o:foo.c

  $(CC) $(CFLAGS) foo.c

bar.o:bar.c

  $(CC) $(CFLAGS) bar.c

实例中,不管全局的$(CFLAGS)的值是什么,在prog以及所引发的所有规则中(prog.o foo.o bar.o),$(CFLAGS)的值都是“-g”

九、模式变量

在GNU的make中,还支持模式变量(Pattern-speific Variable),通过前面,变量可以定义在某个目标上,模式变量的好处就是可以给定一种“模式”,可以把变量定义在符合这种模式的所有目标上。

示例:%.o:CFLAGS=-o

<pattern...>:<variable-assignment>

<pattern...>:override<variable-assignment>

make命令行指定变量override针对系统环境传入变量。