makefile--变量与函数的综合示例

变量与函数的综合示例


1.项目需求

  • 自动生成target文件夹存放可执行文件
  • 自动生成objs文件夹存放编译生成的目标文件(*.o)
  • 支持调试版本的编译选项
  • 考虑代码的扩展性

2. 关键语法

  • $(wildcard _pattern)
    获取当前工作目录中满足_pattern的文件或目录列表
  • $(addprefix _prefix,_names)
    给名字列表_names中的每一个名字增加前缀_prefix

3. 关键技巧

  • 自动获取当前目录下的源文件列表(函数调用)
SRCS += $(wildcard *.c)
  • 根据源文件列表生成目标文件列表(变量的值替换)
OBJS := $(SRCS:.c=.o)
  • 对每一个目标文件列表加上路径前缀(函数调用)
OBJS := $(addprefix path/,$(OBJS))

4. 规则中的模式替换(目录结构)

本质:匹配当前目录下的文件,然后根据模式将当前目录下的文件进行特定替换
前提:工作目录中存在func.c main.c
说明:

  • %.c用于模式匹配当前目录下的文件
  • %.o用于将%.c匹配成功的文件名进行模式替换,.c替换.o
%.o : %.c
    gcc -o $@ -c $^

与变量的值替换的区别:
变量值替换是从给定的列表中将文件一一进行匹配,然后放入指定的依赖中;变量值替换如下所示:

OBJS := func.o main.o
$(OBJS) :  %.o : %.c
    gcc -o $@ -c $^

编译规则的依赖如下:


![10-1.PNG](https://upload-images.jianshu.io/upload_images/10834716-2f45145c35635422.PNG?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

示例代码--变量与函数的综合应用
源代码如下:
const.c

const char* g_hello = "hello-makefile";

func.c

#include "stdio.h"
extern char* g_hello;

void foo()
{
    printf("void foo():%s\n".g_hello);
}   

main.c

extern void foo();

int main()
{
    foo();
    return 0;
}

makefile

CC := gcc
MKDIR := mkdir
RM := rm -rf

DIR_OBJS := objs
DIR_TARGET := target

DIRS := $(DIR_OBJS) $(DIR_TARGET)

#target/hello-makefile.out
TARGET := $(DIR_TARGET/hello-makefile.out
#SRCS = {const.c func.c main.c}
SRCS := $(wildcard *.c)
#OBJS = {const.o func.o main.o}
OBJS := $(SRCS:.c=.o)
#OBJS ={objs/const.o objs/func.o objs/main.o}
OBJS = $(addprefix $(DIR_OBJS)/,$(OBJS))

.PHONY : rebuild clean all

$(TARGET) : $(DIRS) $(OBJS)
    $(CC) -o $@ $(OBJS)
    @echo "Target file => $@"
    

$(DIRS) :
    $(MKDIR) $@
    
$(DIR_OBJS)/%.o : %.c
    ifeq ($(DEBUG),true)
    $(CC) -o $@ -g -c $^
    else
    $(CC) -o $@ -c $^
    endif
    
rebuild : clean all

all : $(TARGET)

clean :
    $(RM) $(DIRS)

10-1.PNG

执行结果如下:

10-2.PNG

下列左图为命令行执行make DEBUG:=true的结果;右图为命令行执行make的结果;可以看到,当命令行指定DEUBG值之后,进入target文件夹后命令行执行objdump -S hello-makefile.out可以看到汇编调试代码中出现了源文件中的代码,方便进行调试。而右图仅仅只有汇编部分代码。

10-3.PNG

小结:

  • 目录可以成为目标的依赖,在规则中创建目录
  • 预定义函数是makefile实战时不可或缺的部分
  • 规则中的模式匹配可以直接针对目录中的文件
  • 可以使用命令行变量编译特殊的目标版本
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 官网 中文版本 好的网站 Content-type: text/htmlBASH Section: User ...
    不排版阅读 4,465评论 0 5
  • 来自陈浩的一片老文,但绝对营养。 示例工程:3 个头文件*.h,和 8 个 C 文件*.c。 初 编译过程,源文件...
    周筱鲁阅读 4,742评论 0 17
  • 常言说三个女人一台戏,我们的办公室里,有十几个女性同事,一天到晚好戏不断。 办公室里有几个正在谈恋爱的小姑娘,隔三...
    黄梅枝阅读 347评论 12 14
  • 感恩我的摩托车失而复得,警察同志给力的追踪,哎呀我的小心脏也刺激了一把,这也许就是我平常保护别人财产的种子开花了吧...
    邬静阅读 106评论 0 1
  • 银河帝国正走向覆亡。 这是一个庞大的帝国,从银河每条巨大旋臂此端至彼端,其间数百万个世界,皆为帝国的势力范围。因而...
    我疯啦我膨胀啦阅读 259评论 0 0