本文已参与「新人创作礼」活动,一起开启掘金创作之路。
1. makefile相关符号
: 依赖开始
<Tab>命令开始
# 注释符号
\ 续行号
% 任意一个
* 所有
? 匹配
[…] 通配符
默认文件名GNUmakefile、makefile、Makefile
1.定义变量
定义 VARNAME=string
使用 ${VARNAME}
Shell的变量用${},不能使用$()访问变量
严格说Makfile的变量用$(),但是Makefile的执行离不开shell环境,
因此使用${}也可以访问,是因为解析的时候使用shell解析.
2.隐含规则
.PHONY 伪目标
$@ 代表目标文件
$^ 代表依赖文件
$< 第一个依赖文件的名称
$+ 所有的依赖文件,以空格分开,并以出现的先后为序,可能包含重复的依赖文件
$? 所有时间戳比目标文件晚的依赖文件,并以空格分开
$* 不包含扩展名的目标文件名称
3.函数
$(函数名 参数1 参数2 参数3 参数4)
示例
1.最初版的makefile:
test:prog.o code.o
gcc –o test prog.o code.o
prog.o:prog.c prog.h code.h
gcc –c prog.c –o prog.o
code.o:code.c code.h
gcc –c code.c –o code.o
clean:
rm –f *.o
2.加了变量后的makefile:
OBJS=prog.o code.o
CC=gcc
test:${ OBJS }
${ CC } –o test ${ OBJS }
prog.o:prog.c prog.h code.h
${ CC } –c prog.c –o prog.o
code.o:code.c code.h
${ CC } –c code.c –o code.o
clean:
rm –f *.o
3.使用隐含规则后的makefile:
OBJS=prog.o code.o
CC=gcc
test:${ OBJS }
${ CC } –o $@ $^
prog.o:prog.c prog.h code.h
code.o:code.c code.h
clean:
rm –f *.o
编译链接过程
shell命令格式:
.c-->.i-->.s--->.o---elf
.h
预编译:gcc -E hello.c -o hello.i
汇编:gcc -S hello.i -o hello.s
编译:gcc -c hello.s -o hello.o
链接:gcc hello.o -o hello
makefile格式:
目标文件:依赖文件
命令
hello:hello.o
gcc hello.o -o hello
hello.i:hello.c
gcc -E hello.c -o hello.i
hello.s:hello.i
gcc -S hello.i -o hello.s
hello.o:hello.s
gcc -c hello.s -o hello.o
清空目标文件的规则
每个Makefile中都应该写一个清空目标文件(.o和执行文件)的规则,这不仅便于重编译,也很利于保持文件的清洁。
一般的风格都是:
clean:
rm edit $(objects)
更为稳健的做法是:
.PHONY : clean
clean :
-rm edit $(objects)
.PHONY意思表示clean是一个“伪目标”,。而在rm命令前面加了一个小减号的意思就是,也许某些文件出现问题,但不要管,继续做后面的事。
当然,clean的规则不要放在文件的开头,不然,这就会变成make的默认目标,相信谁也不愿意这样。不成文的规矩是——“clean从来都是放在文件的最后”。