makefile函数
字符操作函数使用
在 Makefile 中可以使用函数来处理变量,从而让我们的命令或是规则更为的灵活和具有智能。make 所支持的函数也不算很多,不过已经足够我们的操作了。函数调用后,函数的返回值可以当做变量来使用。
函数调用,很像变量的使用,也是以“$”来标识的,其语法如下:
$(\<function\> \<arguments\>)
或是
${\<function\> \<arguments\>}
这里,\<function\> 就是函数名,make 支持的函数不多。\<arguments\> 是函数的参数,参数间以逗号“,”分隔,而函数名和参数之间以“空格”分隔。函数调用以“$”开头,以圆括号或花括号把函数名和参数括起。例如字符替换函数。
字符串替换函数
$(subst \<from\>,\<to\>,\<text\>)
名称:字符串替换函数——subst。
功能:把字串 \<text\> 中的 \<from\> 字符串替换成 \<to\>。
返回:函数返回被替换过后的字符串。
Makefile 内容
all:
@echo $(subst t,e,maktfilt) <-- 将t替换为e
bash 中执行 make
$ make
makefile
模式字符串替换函数
$(patsubst \<pattern\>,\<replacement\>,\<text\>)
名称:模式字符串替换函数——patsubst。
功能:以通配符的形式替换字符,查找 \<text\> 中的单词(单词以“空格”、“Tab”或“回车”“换行”分隔)是否符合模式 \<pattern\>,如果匹配的话,则以 \<replacement\> 替换。这里,\<pattern\> 可以包括通配符“%”,表示任意长度的字串。如果 \<replacement\> 中也包含“%”,那么,\<replacement\> 中的这个“%”将是 \<pattern\> 中的那个“%”所代表的字串。(可以用“\”来转义,以“\%”来表示真实含义的“%”字符)返回:函数返回被替换过后的字符串。
示例:
$(patsubst %.c,%.o,x.c.c bar.c)
把字串“x.c.c bar.c”符合模式 [%.c] 的单词替换成[%.o],返回结果是“x.c.o bar.o”
Makefile 内容
all:
@echo $(patsubst %.c,%.o,programA.c programB.c)
bash 中执行 make
$ make
programA.o programB.o
去首尾空格函数
名称:去空格函数——strip。
功能:去掉 \<string\> 字串中开头和结尾的空字符。
返回:返回被去掉空格的字符串值。
示例:
$(strip a b c)
把字串“a b c ”去到开头和结尾的空格,结果是“a b c”。
Makefile 内容
VAL := " aa bb cc "
all:
@echo "去除空格前: " $(VAL)
@echo "去除空格后: " $(strip $(VAL))
bash 中执行 make
$ make
去除空格前: aa bb cc
去除空格后: aa bb cc
查找字符串函数
$(findstring \<find\>,\<in\>)
名称:查找字符串函数——findstring。
功能:在字串 \<in\> 中查找 \<find\> 字串。
返回:如果找到,那么返回找到的字符串,否则返回空字符串。
示例:
$(findstring a,a b c)
$(findstring a,b c)
第一个函数返回“a”字符串,第二个返回“”字符串(空字符串)
Makefile 内容
VAL := " aa bb cc "
all:
@echo $(findstring aa,$(VAL))
@echo $(findstring ab,$(VAL))
bash 中执行 make
$ make
aa
过滤函数
$(filter \<pattern...\>,\<text\>)
名称:过滤函数——filter。
功能:以 \<pattern\> 模式过滤 \<text\> 字符串中的单词,保留符合模式 \<pattern\> 的单词。可以有多个模式。
返回:返回符合模式 \<pattern\> 的字串。
示例:
sources := foo.c bar.c baz.s ugh.h
foo: $(sources)
cc $(filter %.c %.s,$(sources)) -o foo
$(filter %.c %.s,$(sources))返回的值是“foo.c bar.c baz.s”。
$(filter-out \<pattern...\>,\<text\>)
Makefile 内容
all:
@echo $(filter %.o %.a,program.c program.o program.a)
bash 中执行 make
$ make
program.o program.a
反过滤函数
名称:反过滤函数——filter-out。
功能:以 \<pattern\> 模式过滤 \<text\> 字符串中的单词,去除符合模式 \<pattern\> 的单词。可
以有多个模式。
返回:返回不符合模式 \<pattern\> 的字串。
示例:
objects=main1.o foo.o main2.o bar.o
mains=main1.o main2.o
$(filter-out $(mains),$(objects)) 返回值是“foo.o bar.o”。
Makefile 内容
all:
@echo $(filter-out %.o %.a,program.c program.o program.a)
bash 中执行 make
$ make
program.c
排序函数
名称:排序函数——sort。
功能:给字符串 \<list\> 中的单词排序(升序)。
返回:返回排序后的字符串。
示例:$(sort foo bar lose)返回“bar foo lose” 。
备注:sort 函数会去掉 \<list\> 中相同的单词。
$(word \<n\>,\<text\>)
Makefile 内容
all:
@echo $(sort bac abc acb cab)
bash 中执行 make
$ make
abc acb bac cab
取单词函数
名称:取单词函数——word。
功能:取字符串 \<text\> 中第 \<n\> 个单词。空格隔开(从一开始)
返回:返回字符串 \<text\> 中第 \<n\> 个单词。如果 \<n\> 比 \<text\> 中的单词数要大,那么返回空
字符串。
示例:$(word 2, foo bar baz)返回值是“bar”。
$(wordlist \<s\>,\<e\>,\<text\>)
Makefile 内容
all:
@echo $(word 1,aa bb cc dd)
@echo $(word 5,aa bb cc dd)
@echo $(word 4,aa bb cc dd)
bash 中执行 make
$ make
aa
dd
取单词串函数 wordlist
名称:取单词串函数——wordlist。
功能:从字符串 \<text\> 中取从 \<s\> 开始到 \<e\> 的单词串。\<s\> 和 \<e\> 是一个数字。
返回:返回字符串 \<text\> 中从 \<s\> 到 \<e\> 的单词字串。如果 \<s\> 比 \<text\> 中的单词数要大,那
么返回空字符串。如果 \<e\> 大于 \<text\> 的单词数,那么返回从 \<s\> 开始,到 \<text\> 结束的单
词串。
示例: $(wordlist 2, 3, foo bar baz)返回值是“bar baz”。
Makefile 内容
all:
@echo $(wordlist 1,3,aa bb cc dd)
@echo $(word 5,6,aa bb cc dd)
@echo $(word 2,5,aa bb cc dd)
bash 中执行 make
$ make
aa bb cc
bb
单词个数统计函数 words
$(words \<text\>)
名称:单词个数统计函数——words。
功能:统计 \<text\> 中字符串中的单词个数。
返回:返回 \<text\> 中的单词数。
示例:$(words, foo bar baz) 返回值是“3”。
备注:如果我们要取 \<text\> 中最后的一个单词,我们可以这样:$(word $(words \<text\>
),\<text\> )。
Makefile 内容
all:
@echo $(words aa bb cc dd)
@echo $(words aabbccdd)
@echo $(words )
bash 中执行 make
$ make
4
1
0
首单词函数 firstword
$(firstword \<text\>)
名称:首单词函数——firstword。
功能:取字符串 \<text\> 中的第一个单词。
返回:返回字符串 \<text\> 的第一个单词。
示例:$(firstword foo bar)返回值是“foo”。
备注:这个函数可以用 word 函数来实现:$(word 1,\<text\>)。
小结
以上,是所有的字符串操作函数,如果搭配混合使用,可以完成比较复杂的功能。这里,举一个现实中应用的例子。我们知道,make 使用“VPATH”变量来指定“依赖文件”的搜索路径。于是,我们可以利用这个搜索路径来指定编译器对头文件的搜索路径参数 CFLAGS,
如将头文件路径的前面加上 - I 表示包含路径:
override CFLAGS += $(patsubst %,-I%,$(subst :, ,$(VPATH)))
如果我们的“$(VPATH)”值是“src:../headers”,VPATH 之间用分号 {:} 隔开,$(subst :, ,$(VPATH)) 先将路径中的分号替换为空格 src ../headers,然后模式替换函数 $(patsubst %,-I%,)在每个路径前面加上 -I,表示包含文件路径,生成 “-Isrc -I../headers”,这正是 cc 或 gcc 搜索头文件路径的参数。