变量与函数的综合示例
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 $^
复制代码
编译规则的依赖如下:
示例代码--变量与函数的综合应用 源代码如下: 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)
复制代码
执行结果如下:
下列左图为命令行执行make DEBUG:=true的结果;右图为命令行执行make的结果;可以看到,当命令行指定DEUBG值之后,进入target文件夹后命令行执行objdump -S hello-makefile.out可以看到汇编调试代码中出现了源文件中的代码,方便进行调试。而右图仅仅只有汇编部分代码。
小结:
- 目录可以成为目标的依赖,在规则中创建目录
- 预定义函数是makefile实战时不可或缺的部分
- 规则中的模式匹配可以直接针对目录中的文件
- 可以使用命令行变量编译特殊的目标版本