在上图可以看到变量 ceshi2 的值是 temp,也就是变量 ceshi1 最后一次的赋值。
2. “ := ” 赋值符
我们修改“=”赋值符中的代码,第二行的“=”改成“:=”,代码如下:
1 ceshi1 = test
2 ceshi2 := $(ceshi1)
3 ceshi1 = temp
4
5 out:
6 @echo ceshi2:$(ceshi2)
我们在终端输入“make out”命令,如下图所示:
file:///C:UsersADMINI~1AppDataLocalTempksohtml10068wps4.png
我们可以看到上图的运行结果输出变量 ceshi2 的值是 test,虽然在第三行我们修改了变量 ceshi1 的
值,通过本实验我们可以看到“:=”赋值符的功能了。
3. “ ?= ” 赋值符
ceshi ?= test
“?=”赋值符的作用是如果前面没有给变量 ceshi 赋值,那么变量就赋值“test”,如果前面已经赋值了,就使用前面的赋值。
4. “ += ” 赋值符
objs = main.o
objs += calc.o
上面的脚本最后变量 objs 的值是“main.o calc.o”,“+=”赋值符的功能是实现变量的追加。
3.4.3 条件判断
使用条件判断,可以让 make 根据运行时的不同情况选择不同的执行分支。条件表达式可以是比较变量的值,或是比较变量和常量的值。其语法有下面两种:
1.
<条件比较>
[条件为真时执行的脚本]
endif
2.
<条件比较>
[条件为真时执行的脚本]
else
[条件为假时执行的脚本]
endif
条件比较用到的比较关键字有:ifeq、ifneq、ifdef、ifndef。
ifeq 表示如果比较相等,语法如下:
ifeq(<参数 1>, <参数 2>)
ifneq 表示如果不相等,语法如下:
ifneq(<参数 1>, <参数 2>)
ifdef 表示如果定义了变量,语法如下:
ifdef <变量名>
ifndef 表示如果没有定义变量,语法如下:
ifndef <变量名>
3.4.4 使用函数
在 Makefile 中可以使用函数来处理变量,从而让我们的命令或是规则更为的灵活和具有智能。make 所支持的函数也不算很多,不过已经足够我们的操作了。函数调用后,函数的返回值可以当做变量来使用。
函数的调用很像变量的使用,也是以“$”来标识的,语法如下:
$(<函数名> <参数集合>)
或者:
${<函数名> <参数集合>}
函数名和参数集合之间以空格分隔,参数集合的参数通过逗号分隔。函数调用以“$”开头,以圆括号或花括号把函数名和参数括起。感觉很像一个变量。函数中的参数可以使用变量。为了风格的统一,函数和变量的括号最好一样,如使用“$(subst a,b,$(x))”这样的形式,而不是“$(subst a,b,${x})”的形式。
因为统一会更清楚,也会减少一些不必要的麻烦。
接下来我们介绍几个常用的函数,其它的函数可以参考文档《跟我一起写 Makefile》。
t 1.subst 函数
$(subst ,,)
此函数的功能是把字串中的字符串替换成,函数返回被替换过后的字符串。如下示例:
$(subst ee,EE,feet on the street)
以上脚本实现把字符串“feet on the street”中的“ee”字符串替换成“EE”字符串,替换后的字符串
为“feet on the strEEt”。
. 2. t patsubst 函数
此函数的功能是查找中的单词(单词以“空格”、“Tab”或“回车”“换行”分隔)是否符合模式,如果匹配的话,则以替换。这里可以包括通配符“%”,表示任意长度的字串。如果中也包含“%”,那么中的这个“%”将是中的那个“%”所代表的字串。(可以用“”来转义,以“%” 来表示真实含义的“%”字符)。函数返回被替换过后的字符串。如下示例: $(patsubst %.c,%.o,x.c bar.c)
以上脚本实现把字串“x.c bar.c”符合模式[%.c]的单词替换成[%.o],返回结果是“x.o bar.o”
3.strip 函数
$(strip )
此函数的功能是去掉字串中开头和结尾的空字符,函数返回被去掉空格的字符串值。如下示例:
$(strip a b c )
以上脚本实现把字串“a b c ”去掉开头和结尾的空格,结果是“a b c”。
. 4. g findstring 函数
$(findstring ,)
此函数的功能是在字串中查找字串,如果找到,那么返回,否则返回空字符串,如下示
例:
$(findstring a,a b c)
$(findstring a,b c)
以上脚本,第一个返回“a”字符串,第二个返回空字符串。
r 5.dir 函数
$(dir )
此函数的功能是从文件名序列中取出目录部分。目录部分是指最后一个反斜杠(“/”)之前的部
分。如果没有反斜杠,那么返回“./”。返回文件名序列的目录部分,如下示例:
$(dir src/foo.c hacks)
以上脚本运行结果返回“src/”。
. 6. r notdir 函数
$(notdir )
此函数的功能是从文件名序列中取出非目录部分。非目录部分是指最后一个反斜杠(“/”)之后
的部分,返回文件名序列的非目录部分,如下示例:
$(notdir src/foo.c)
以上脚本返回字符串“foo.c”
. 7. h foreach 函数
$(foreach ,,)
此函数的功能是把参数中的单词逐一取出放到参数所指定的变量中,然后再执行所包含的表达式。每一次会返回一个字符串,循环过程中,的所返回的每个字符串会以空格分隔,最后当整个循环结束时,所返回的每个字符串所组成的整个字符串(以空格分隔)将会是 foreach 函数的返回值。所以,最好是一个变量名,可以是一个表达式,而中一般会使用这个参数来依次枚举中的单词。如下示例:
names := a b c d
files := $(foreach n,$(names),$(n).o)
以上脚本实现$(name)中的单词会被挨个取出,并存到变量“n”中,“$(n).o”每次根据“$(n)”计算出一个值,这些值以空格分隔,最后作为 foreach 函数的返回,所以$(files)的值是“a.o b.o c.o d.o”。(注意,foreach 中的参数是一个临时的局部变量,foreach 函数执行完后,参数的变量将不在作用,其作用域只在 foreach 函数当中)。
3.4.5 在规则中使用通配符
如果我们想定义一系列比较类似的文件,我们很自然地就想起使用通配符。make 命令支持三种通配符:“*”,“?”和“[...]”,这是和 Unix 的 B-Shell 是相同的。“~”字符在文件名中也有比较特殊的用途。如果是“~/test”,这就表示当前用户根目录下的 test 文件。而“~admin/test”则表示用户 admin 根目录下的 test 文件。通配符代替了一系列的文件,如“*.c”表示所有后缀为 .c 的文件。一个需要我们注意的是,如果我们的文件名中有通配符,如:“*”,那么可以用转义字符“”,如“*” 来表示真实的“*”字符,而不是任意长度的字符串。
下面我们来看几个具体的示例:
clean:
rm -rf *.o
上面这个示例说明通配符可以在规则的命令中使用。
print: *.c
上面这个示例说明通配符可以在规则的依赖中使用
objects = *.o
上面这个示例表示了,通符同样可以用在变量中。并不是说[*.o]会展开,objects 的值就是“*.o”。Makefile中的变量其实就是 C/C++中的宏。如果你要让通配符在变量中展开,也就是让 objects 的值是所有[.o]的文件名的集合,那么,你可以这样:
objects := $(wildcard *.o)
这种用法由关键字“wildcard”指出,关于 Makefile 的关键字可以参考文档《跟我一起写 Makefile》。
关于 Makefile 的相关内容我们就介绍到这里,本节只是对 Makefile 做了基本的讲解,Mkaefile 还有大量
完结,更多内容关注: