make之eval函数

函数原型:
$(eval text)
它的意思是 text 的内容将作为makefile的一部分而被make解析和执行。
需要注意的是该函数在执行时会对它的参数进行两次展开,第一次展开是由函数本身完成,第二次是函数展开后的结果被作为makefile内容时由make解析时展开.

1.函数二次展开示例

define MA
aa:aa.c
    gcc -g -o aa aa.c
endef

$(eval $(call MA) )

会产生一个这样的编译:gcc -g -o aa aa.c

2.模板示例

2.1 模板引出示例

PROGRAMS := SERVER CLIENT

SERVER_OBJS := server.o 

CLIENT_OBJS := client.o 

.PHONY: all
all: $(PROGRAMS)

SERVER: $(SERVER_OBJS)
ALL_OBJS += $(SERVER_OBJS)

$(info $(ALL_OBJS))

CLIENT: $(CLIENT_OBJS)
ALL_OBJS += $(CLIENT_OBJS)

$(info $(ALL_OBJS))

#从上面可以看出,生成SERVER和CLIETN的两个规则基本上是一样的,那么可以使用更通用的规则表示

$(PROGRAMS):
    gcc $^ -o $@

2.2 call与eval组合注意事项

image

image

第8行 引用变量SERVER_OBJS前多了一个空格,显然是引用不到server.o
第9行 可以正常引用
第20,25行 由于29行eval还没有执行,所以为空
第27行 打印函数call调用后的返回值,可以看到依赖展开后 中间多了一个空格

2.3 call与eval组合完整示例

CC=gcc
PROGRAMS=SERVER  CLIENT

SERVER_OBJS:=server.o 

CLIENT_OBJS:=client.o 

.PHONY: all
all: $(PROGRAMS)

#define PROGRAM_TEMPLATE
#$(1):$$($(1)_OBJS)
#ALL_OBJS += $$($(1)_OBJS)
#endef

define PROGRAM_TEMPLATE
$(strip $(1)):$$($(strip $(1))_OBJS)
ALL_OBJS += $$($(strip $(1))_OBJS)
endef

#$(info 111$(ALL_OBJS))

#下面两行的区别在于call函数中 逗号和参数[$(prog)]之间的空格.
#因此,PROGRAM_TEMPLATE更好的写法是去掉参数前面的空格,如果有的话
$(foreach prog, $(PROGRAMS), $(eval $(call PROGRAM_TEMPLATE, $(prog))))
#$(foreach prog, $(PROGRAMS), $(eval $(call PROGRAM_TEMPLATE,$(prog))))

#$(info 222-$(ALL_OBJS))

$(PROGRAMS):
    gcc $^ -o $@

clean:
    rm -f $(ALL_OBJS) $(PROGRAMS)


©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 来自陈浩的一片老文,但绝对营养。 示例工程:3 个头文件*.h,和 8 个 C 文件*.c。 初 编译过程,源文件...
    周筱鲁阅读 10,148评论 0 17
  • 你不迈开脚步去“看看”这个偌大而多彩的世界,你永远都不会知道这个世界是怎样的一种情况,你永远都不会知道自己的世界有...
    洛铁花开阅读 4,119评论 3 2
  • 以前总是不明白爸妈口中的“学习是为了你自己,不是为了我们”是什么意思,但是好像某一个瞬间我突然明白了,当你太爱他们...
    我是露西阅读 2,118评论 0 0
  • 案主:男性,26岁,大专,自主创业 案主自述:画面感是想表达空灵,投入 我想你大概是一个比较细腻的人,喜欢有...
    蓝月菁芸阅读 1,644评论 0 0