Makefile中的常用函数 - foreach、if、call、shell、value、eval

foreach循环函数

foreach是Makefile中用来做循环的函数,它把可以重复利用一段脚本,但是每次又有不同的条件。它类似于Unix标准Shell/bin/sh)中的for语句,或是C-Shell/bin/csh)中的foreach语句。它的语法是:

       $(foreach  var, list, text)

       前两个参数var和list,参数中的单词逐一取出放到参数所指定的变量中,然后再执行所包含的表达式。每一次会返回一个字符串,循环过程中,的所返回的每个字符串会以空格分隔,最后当整个循环结束时,所返回的每个字符串所组成的整个字符串(以空格分隔)将会是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函数当中。

if 函数

if函数很像GNUmake所支持的条件语句——ifeq(参见前面所述的章节),if函数的语法是:

       $(if ,)

 或是

       $(if ,,)

可见,if函数可以包含“else”部分,或是不含。即if函数的参数可以是两个,也可以是三个。参数是if的表达式,如果其返回的为非空字符串,那么这个表达式就相当于返回真,于是,会被计算,否则会被计算。

if函数的返回值是,如果为真(非空字符串),那个会是整个函数的返回值,如果为假(空字符串),那么会是整个函数的返回值,此时如果没有被定义,那么,整个函数返回空字串。

所以,只会有一个被计算。

 
      补充一个ifeq的用法:
Makefile中的ifeq定义前面不能被tab制表符修饰,只能被空格。这与Makefile函数正好是相反的。
当出现语法错误时,务必检查是否在ifeq前面有tab制表符    

/bin/sh: -c: line 0: syntax error near unexpected token `nightly,nightly'

/bin/sh: -c: line 0: `ifeq (nightly,nightly)'


ifeq ($(1), arg1)
    # text do something
endif


call函数

call函数是唯一一个可以用来创建新的参数化的函数。你可以写一个非常复杂的表达式,这个表达式中,你可以定义许多参数,然后你可以用call函数来向这个表达式传递参数。其语法是:

      $(call ,,,...)

 make执行这个函数时,参数中的变量,如$(1)$(2)$(3)等,会被参数依次取代。而的返回值就是call函数的返回值。例如:
       reverse = $(1) $(2)

       foo = $(call reverse, a, b)

那么,foo的值就是“a b”。当然,参数的次序是可以自定义的,不一定是顺序的,如:

       reverse = $(2) $(1)

       foo = $(call reverse,a,b)

此时的foo的值就是“b a”

shell函数

shell函数也不像其它的函数。顾名思义,它的参数应该就是操作系统Shell的命令。它和反引号“`”是相同的功能。这就是说,shell函数把执行操作系统命令后的输出作为函数返回。于是,我们可以用操作系统命令以及字符串处理命令awksed等等命令来生成一个变量,如:

      contents := $(shell cat foo)

      files := $(shell echo *.c)

注意,这个函数会新生成一个Shell程序来执行命令,所以你要注意其运行性能,如果你的Makefile中有一些比较复杂的规则,并大量使用了这个函数,那么对于你的系统性能是有害的。特别是Makefile的隐晦的规则可能会让你的shell函数执行的次数比你想像的多得多。


value函数

       value函数提供了一种使用变量非展开值的方式,但是它也并不能对于已经在定义中展开的结果生效。比如,你使用$()定义了一个变量,那么就可以用value来使用变量的非展开值。value的语法是:

       $(value  var)

       注意,这里的var参数是变量名,而不是对变量的引用$(var)

       例子:

       FO=$(PATH)

       all:

              echo 1$(FO)

              echo 2 $(value FO)

       输出是1 /bin:/usr/bin:/usr/local/sbin:/usr/sbin:/sbin:   (系统PATH的展开)

2 $(PATH),FOO的定义。

       Value函数经常与eval函数一起结合使用。


eval函数

       eval函数可以通过其他变量或函数的结果来定义新的makefile结构。Eval的参数被按照makefile的语法格式展开。展开的结果可以用来构造新的变量、目标、隐式或者显式规则。

       eval的参数被展开两次,第一次被eval本身,第二次是第一次展开的结果被解析为makefile语法时。因此在使用eval时需要提供额外的$引用字符。这时候也可以使用value来避免不希望的展开。

       下面是一个用eval来创建目标的例子,尽管看起来有些复杂,但是在写好它之后,就可以把generic的部分提供到一个公共文件中,再在你的个人makefile中include它,这样子你的个人makefile就会特别简练了。

  1. PROGRAMS = server client
  2. server_OBJS = server.o server_priv.o server_access.o
  3. server_LIBS = priv protocol
  4. client_OBJS = client.o client_api.o client_mem.o
  5. client_LIBS = protocol

  6. # Everything after this is generic,可以拿到其他文件用来include
  7. .PHONY: all
  8. all: $(PROGRAMS)

  9. define PROGRAM_template =
  10. $(1): $$($(1)_OBJS) $$($(1)_LIBS:%=-l%)
  11. ALL_OBJS += $$($(1)_OBJS)
  12. endef

  13. $(foreach prog,$(PROGRAMS),$(eval $(call PROGRAM_template,$(prog))))
  14. $(PROGRAMS):
  15. $(LINK.o) $^ $(LDLIBS) -o $@
  16. clean:
  17. rm -f $(ALL_OBJS) $(PROGRAMS)

     更多的eval函数实用例子可见 OpenWRT的package编译文件,里面大量的使用了eval模板来构建target

  • 0
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: makefileforeach用法是用来遍历一个列表的每个元素,并对每个元素执行相同的操作。其语法如下: $(foreach var, list, text) 其,var是一个变量名,list是一个以空格分隔的列表,text是要执行的操作。 例如,假设我们有一个列表包含三个文件名:file1.c、file2.c和file3.c,我们想要对每个文件执行编译操作,可以使用foreach: $(foreach file, file1.c file2.c file3.c, gcc -c $(file)) 上述语句将会执行以下三个操作: gcc -c file1.c gcc -c file2.c gcc -c file3.c 这样,我们就可以方便地对一个列表的每个元素执行相同的操作。 ### 回答2: makefileforeach用法是一个用于遍历列表的循环语句。它可以通过指定变量和列表,对列表的每个元素执行相同的操作。具体用法如下: foreach 变量, 列表 <操作> endef 其,变量是一个临时变量名,用于保存列表的每个元素,并在循环被使用。列表是一个包含多个元素的字符串,每个元素之间使用空格或逗号分隔。操作是在循环需要执行的命令或规则。 在循环,变量会依次取得列表的每个元素的值,并将其赋给变量。然后,循环会执行操作部分的命令或规则。在每次循环结束后,变量会被重新赋值为下一个元素的值,直到所有元素遍历完毕。 使用foreach的一个例子如下: files := file1.c file2.c file3.c $(foreach file,$(files),\ $(info Compiling $(file))\ $(CC) -c $(file) -o $(basename $(file)).o\ ) 上面的例子,我们首先定义了一个变量files,它包含了三个文件名。然后使用foreach循环,每次将一个文件名赋给变量file,并执行info命令打印出正在编译的文件名,并执行编译命令将该文件编译成对应的.o文件。 总之,makefileforeach用法是一个非常有用的循环语句,可以方便地遍历列表的元素,并执行相应的操作。 ### 回答3: makefileforeach是一个循环语句,用于对指定的变量列表的每个元素执行相同的操作。 其语法如下: foreach <var>,<list>,<text> <text> endef 其,<var>是一个指定的变量名,<list>是一个用空格分隔的变量列表,<text>是在每个列表元素上执行的命令。 当make执行到foreach语句时,会将<list>的每个元素赋值给<var>,然后将<text>的命令依次执行。 例如,假设有一个变量列表src_files,其包含了多个源文件名,我们需要对每个文件执行编译操作。可以使用foreach语句来实现: src_files := file1.c file2.c file3.c compile: $(foreach file,$(src_files),gcc -c $(file);) 在上述例子foreach循环会将src_files的每个文件名依次赋值给file变量,并执行gcc -c命令进行编译操作。 值得注意的是,在foreach语句执行的命令必须以分号结尾,并且需要使用$()语法来引用变量。 使用foreach语句可以简化对变量列表每个元素执行相同操作的情况,提高makefile的可读性和灵活性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值