Makefile目标,伪目标,头文件自动依赖
目标
即我们最终要生成的文件,make默认生成第一个目标,注意 makefile中tab
和空格不是一回事,规则使用tab
缩进,编辑器不要设置诸如“将tab替换为空格之类的选项”,目标的结构如下
目标:依赖1 依赖2
<TAB>生成目标的规则
伪目标
无论如何也要执行的目标,需要使用.PHONY:
标识
.PHONY:clean
clean:
rm -f *.o
常见伪目标
all
是所有目标的伪目标,功能是编译所有目标
clean
删除所有被make创建的文件
install
安装已经编译好的程序,其实就是把目标执行文件拷贝到指定目标中
print
这个伪目标的功能是列出改变过的源文件
tar
把源程序打包备份,就是一个tar文件
dist
创建一个压缩文件,一般吧tar文件压缩成Z文件或gz文件
TAGS
更新所有目标,以备完整的重编译使用
check/test
测试makefile流程
例子
circle:main.o circle.o
cc main.o circle.o -o circle
main.o:main.c
cc -c main.c
circle.o :circle.c
cc -c circle.c
clean:
rm *.o *.h circle
自动依赖头文件
makefile是根据依赖关系,时间戳和生成规则来判断哪些文件需要更新,但是我们通常写的生成规则并不会包含头文件,依赖关系也不会,这样如果我们更新了头文件,makefile也不会发现,也就不会更新相应的文件,同时,考虑到头文件包含关系的复杂性,我们在写依赖关系之前把所有的头文件理清楚再写进去也不现实,一个好的方式是利用$gcc -M
或$gcc -MM
功能,这两个功能能自动的分析头文件的依赖关系,前者分析目标文件的所有头文件依赖关系,后者分析和我们自定义文件的依赖关系:
$gcc -M hello.c
hello.o: hello.c /usr/include/stdio.h /usr/include/features.h \
/usr/include/i386-linux-gnu/bits/predefs.h \
/usr/include/i386-linux-gnu/sys/cdefs.h \
/usr/include/i386-linux-gnu/bits/wordsize.h \
/usr/include/i386-linux-gnu/gnu/stubs.h \
/usr/include/i386-linux-gnu/gnu/stubs-32.h \
/usr/lib/gcc/i686-linux-gnu/4.6/include/stddef.h \
/usr/include/i386-linux-gnu/bits/types.h \
/usr/include/i386-linux-gnu/bits/typesizes.h /usr/include/libio.h \
/usr/include/_G_config.h /usr/include/wchar.h \
/usr/lib/gcc/i686-linux-gnu/4.6/include/stdarg.h \
/usr/include/i386-linux-gnu/bits/stdio_lim.h \
/usr/include/i386-linux-gnu/bits/sys_errlist.h hello.h
$gcc -MM hello.c
hello.o: hello.c hello.h
可以看到,他们的输出的格式刚好就是我们写的依赖关系的格式,所以我们可以综合利用伪目标,重定向,include
关键字实现对头文件更新的自动检查
.PHONY:all
all:hello dependency
hello:hello.o
gcc hello.o -o hello
hello.o:hello.c
gcc -c hello.c -o hello.o
dependency:headers
headers:
gcc -M hello.c > headers
include headers
.PHONY:clean
clean:
rm -rf hello *.o
分析:make缺省目标会自动以第一个目标为目标,首先依次检查hello<-hello.o<-hello.c发现没有文件被更新,dependency并不是一个文件,也没有时间戳,所以也没有被更新,但是dependency每次都会执行刷新headers文件,我们又include了headers,所以一旦headers里面写的头文件有更新,就会通过hello.o<-xxx.h来找到更新关系,进而更新hello.o并进一步更新hello。至此实现了对头文件的自动依赖问题的解决。
执行效果:
//程序就是输出这个N值
#ifndef __HELLO_H__
#define __HELLO_H__
#define N 121220;
#endif
$./hello
num:121220
$vi hello.h #修改头文件中N的值为5
$make #果然重新编译了
gcc -c hello.c -o hello.o
gcc hello.o -o hello
gcc -M hello.c > headers
$./hello
num:5