学习自狄泰软件学院唐佐临老师Makefile课程,文章中图片取自老师的PPT,仅用于个人笔记。
规则中的模式替换
此处的规则中的模式替换针对的是当前目录中的文件,是直接进行模式替换而第六课中学习的规则中的模式替换,针对的是变量(保存着文件列表的变量),如下:
我们在第六课中学习过 规则中的模式替换,和这里学习的不一样,第六课中学习的规则中的模式替换的 目标的写法是 :
$(OBJS) : %.o
而 我们第十课这里学习的目标的写法是这样的 :
%.o : %.c
这两个规则中的模式替换写法是不一样的,第六课中的目标是从 $(OBJS) 这个 变量所对应的列表中产生的,即 $(OBJS) : %.o 模式匹配,把 变量 $(OBJS) 里面符合 “%.o” 这个模式的目标名取出来作为规则目标,然后向后进行,将取出来的目标名的后缀换成.c 作为依赖。所以 第六课中学习的规则中的模式替换 所针对的目标是一个保存文件名列表的变量,即针对的是一个列表
而第十课中的规则中模式替换,是直接对当前工作目录中的 文件进行模式替换,逐个的对工作中的文件来进行替换。第十课中的规则中的模式替换是针对当前工作目录中的文件的模式替换
第六课中的 规则中的模式替换
实验1
CC := gcc
MKDIR := mkdir
RM := rm -fr
DIR_OBJS := objs
DIR_TARGET := target
#辅助变量
DIRS := $(DIR_OBJS) $(DIR_TARGET)
#在 可执行该文件 前加上了路径前缀 target/hello-makefile.out
TARGET := $(DIR_TARGET)/hello-makefile.out
# 获取当前目录的所有.c 文件的文件名列表 : main.c const.c func.c
SRCS := $(wildcard *.c)
# 通过变量的值替换得到所有源文件对应的目标文件的文件名列表 :main.o const.o func.o
OBJS := $(SRCS:.c=.o)
# 在各个目标文件的文件名前面加上路径名前缀 : objs/main.o objs/const.o objs/func.o
OBJS := $(addprefix $(DIR_OBJS)/, $(OBJS))
.PHONY : rebuild clean all
#定义生成可执行文件 target/hello-makefile.out 的规则,依赖于 objs,target 两个文件夹 以及 一系列的目标文件 main.o const.o func.o
# $(DIRS) $(OBJS) 这两个依赖,对应着有他们各自的规则
$(TARGET) : $(DIRS) $(OBJS)
# 链接 gcc -o target/hello-makefile.out objs/main.o objs/const.o objs/func.o
$(CC) -o $@ $(OBJS)
@echo "Target File ==> $@"
# 文件夹对应的规则
$(DIRS) :
$(MKDIR) $@
#目标文件对应的规则,基于目录结构的 规则中的模式替换:
#将 当前工作目录的每一个c文件 后缀 替换为 .o 然后作为目标,最后在.o文件前面加上路径前缀
$(DIR_OBJS)/%.o : %.c
$(CC) -o $@ -c $^
rebuild : clean all
all : $(TARGET)
clean :
$(RM) $(DIRS)
mhr@ubuntu:~/work/makefile1$ make all
mkdir objs //创建目标文件夹
mkdir target //创建可执行文件文件夹
gcc -o objs/const.o -c const.c // 生成目标文件
gcc -o objs/main.o -c main.c
gcc -o objs/func.o -c func.c
gcc -o target/hello-makefile.out objs/const.o objs/main.o objs/func.o // 链接目标文件成可执行文件
Target File ==> target/hello-makefile.out
mhr@ubuntu:~/work/makefile1$
实验2 :添加编译选项
CC := gcc
MKDIR := mkdir
RM := rm -fr
DIR_OBJS := objs
DIR_TARGET := target
DIRS := $(DIR_OBJS) $(DIR_TARGET)
TARGET := $(DIR_TARGET)/hello-makefile.out
# main.c const.c func.c
SRCS := $(wildcard *.c)
# main.o const.o func.o
OBJS := $(SRCS:.c=.o)
# objs/main.o objs/const.o objs/func.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)
mhr@ubuntu:~/work/makefile1$
mhr@ubuntu:~/work/makefile1$ make DEBUG:=true
mkdir objs
mkdir target
gcc -o objs/const.o -g -c const.c
gcc -o objs/main.o -g -c main.c
gcc -o objs/func.o -g -c func.c
gcc -o target/hello-makefile.out objs/const.o objs/main.o objs/func.o
Target File ==> target/hello-makefile.out
mhr@ubuntu:~/work/makefile1$