变量与函数的综合示例
1、实战需求
- 自动生成target文件夹存放可执行文件
- 自动生成obj文件夹存放编译生成的目标文件(*.o)
- 支持调试版本的编译选项
- 考虑代码的扩展性
2、工具原料
$(wildcard _pattern)
- 获取当前工程目录中满足
_pattern
的文件或目录列表
- 获取当前工程目录中满足
$(addprefix _prefix, _names)
- 给名字列表
_names
中的每一个名字增加前缀_prefix
- 给名字列表
3、关键技巧
- 自动获取当前目录下的源文件列表(函数调用)
SRCS := $(wildcard *.c)
- 根据源文件列表生成目标文件列表(变量的值替换)
OBJS := $(SRCS:.c=.o)
- 对每一个目标文件列表加上路径前缀(函数调用)
OBJS := $(addprefix path/,$(OBJS))
- 规则中的模式替换(目录结构)
对比第6课中,
4、编译规则的依赖
5、编程实验
#重新定向变量
CC := gcc
MKDIR := mkdir
RM := rm -fr
#定义2个目录
DIR_OBJS := objs
DIR_TARGET := target
#目录列表
DIRS := $(DIR_OBJS) $(DIR_TARGET)
#定义生成目标
TARGET := $(DIR_TARGET)/hello-makfile.out
#main.c func.c const.c
SRCS := $(wildcard *.c)
#main.o func.o const.o
OBJS := $(SRCS:.c=.o)
#objs/main.o objs/func.o objs/const.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
mkdir objs
mkdir target
gcc -o objs/const.o -g -c const.c
DEBUG ==> true
gcc -o objs/main.o -g -c main.c
DEBUG ==> true
gcc -o objs/fun.o -g -c fun.c
DEBUG ==> true
gcc -o target/hello-makfile.out objs/const.o objs/main.o objs/fun.o
Target File ==> target/hello-makfile.out
6、小结
- 目标可以成为目标的依赖,在规则中创建目录
- 预定义函数是makefile实战开发不可或缺的部分
- 规则中的模式匹配可以直接针对目录中的文件
- 可以使用命令行变量编译特殊的目标版本