Makefile
1. gcc -o test a.c b.c
缺点:对所有文件都再处理一次,应该分别编译,最后链接
gcc –c –o a.o a.c (分别编译)
gcc –c –o b.o b.c
gcc –o test a.o b.o (最后链接)
如何判断哪个文件被修改?比较时间
Makefile最根本的元素:规则
规则
目标文件:依赖文件
Tab键 命令
规则:当“依赖”比“目标”新或者目标文件不存在的时候,就会执行命令
Makefile:名字为makefile的文件放置以下内容,想执行以下命令时,使用make命令
test:a.o b.o
gcc –o test a.o b.o
a.o:a.c
gcc –c –o a.o a.c
b.o:b.c
gcc –c –o b.o b.c
第一次执行make,三条命令都执行,
修改a.c ,第二次执行make只会执行
//test:a.o b.o
gcc –o test a.o b.o
//a.o:a.c
gcc –c –o a.o a.c
2.Makefile的语法
(1)Makefile通配符
test:a.o b.o c.o d.o(所依赖的目标文件可以在这里增加即可)
gcc –o test a.o b.o c.o ========》gcc –o test $^ ;$^代表所有的依赖
a.o:a.c========》%.o:%.c
gcc –c –o a.o a.c========》gcc –c –o $@ $<
//这里的$@代表目标文件 ;$<代表依赖文件
(2)假想目标
test:a.o b.o c.o
gcc –o test $^
%.o:%.c
Gcc –c –o $@ $<
clean:
rm *.o test
如何使用Makefile:make 目标;若无目标,默认只得到第一个目标;若有目标,则得到目标
Make clean 执行一次后,会得到clean文件,则无法第二次执行make clean 因为没有依赖文件做比较,解决办法:将clean定义为假想目标
.PHONY clean
(3)Makefile的变量:即时变量 延时变量 export
即时变量(简单变量)比如:A:=XXX #A的值即可确定
延时变量 B=XXX #B的值使用到时才确定
执行结果: A= B=abc
?= #延时变量 如果是第一次定义,起作用,如果在前面已经定义过,则忽略此句
3.Makefile函数
(1)$(foreach var,list, text)对于list中的每一个变量var都执行text的内容
A =a b c
B=$(foreach f,$(A),$(f).o) //对A里面的每个参数 执行$(f).o操作
all:
@echo B=$(B)
输出结果为:B=a.o b.o c.o
(2)$(filter pattern…,text) 在text中取出符合pattern格式的值
$(filter-out pattern…,text) 在text中取出符合pattern格式的值
输出结果:D=d/ E= a b c
(3)$(patsubst 原模式, 目标模式, 文件列表)
如果文件列表里取出每一个值,如果符合符合原模式,则替换为目标模式
4.
gcc –M c.c会生成c.c的依赖文件并打印出
gcc –M –MF c.d c.c 把生成的c.c的依赖文件放入c.d文件存储
gcc-c –o c.o c.c –MD –MF c.d把依赖文件写入c.d 的同时编译c.c生成c.o
objs = a.o b.o c.o
dep_files := $(patsubst %,.%.o.d,$(objs)) //每个文件都生成自己的依赖文件
dep_files := $(wildcard $(dep_files)) //取出真正存在的依赖文件
CFLAGS = -Werror –I. //把所有的warn都当做错误编译器在指定当前目录下查找
// CFLAGS = -Werror –Iinclude //指定include作为当前h文件搜索的默认路径
test:$(objs)
gcc –o test $^
ifneq ($(dep_files))//如果dep_files不为空
#include ($(dep_files))
endif
%.o:%.c
gcc $(CFLAGS) –c –o $@ $< –MD –MF .$@.d //这句话生成所有.c文件的依赖文件.c.o.d
clean:
rm *.o test
distclean:
rm $( dep_files)
.PHONY: clean