2006-11-13 09:43:57| 分类: 默认分类 |字号 订阅
从网上学到的makefile最基本的知识:
这里先介绍makefile最基本的知识。
假设现有三个 c 程序 main.c, mytool1.c, mytool2.c, 其中主程序在 main.c中。
gcc -c main.c
gcc -c mytool1.c
gcc -c mytool2.c
gcc -o main main.o mytool1.o mytool2.o
这样的话我们可以产生main程序,而且也不是很麻烦.但是我们考虑一下,如果有一天我们修改了其中的一个文件(比如说mytool1.c),那么我们难道还要重新输入上面的命令吗?也许你会说,这个很容易解决啊,我写一个SHELL脚本,让它帮我去完成不就可以了.是的对于这个程序来说,是可以起到作用的.但是当我们把事情想的更复杂一点,如果我们的程序有几百个源程序的时候,难道也要编译器重新一个一个的去编译吗?
为此,聪明的程序员们想出了一个很好的工具来做这件事情,这就是make.
什么是make
linux系统中,提供了一个自动生成和维护目标程序的工具---make。make是一个单独工作的程序,根据程序模块的修改情况重新编译链接目标代码,以保证目标代码总是由最新的模块组成。
源程序编译链接的顺序和范围取决于模块之间的关联关系。make 从指定文件中读取说明模块之间关系的信息,然后根据这些信息维护和更新目标文件。具体地说,make 首先判断哪些文件过时了。所谓过时,就是一个文件生成以后,用来生成该文件的源文件或者所依赖的模块被修改了,而他自己没有修改。其判断依据是文件生成时间。找到过时的文件,就根据指定文件中的有关信息进行更新操作。
make 默认从GNUmakefile makefile 或者 Makefile 读取信息。可以用 -f 指定文件。
我们只要执行以下make,就可以把上面的问题解决掉.在我们执行make之前,我们要先编写一个非常重要的文件.--Makefile(或 makefile).对于上面的那个程序来说,可能的一个Makefile的文件是:
#这是上面那个程序的Makefile文件
main:main.o mytool1.o mytool2.o
gcc -o main main.o mytool1.o mytool2.o
main.o:main.c mytool1.h mytool2.h
gcc -c main.c
mytool1.o:mytool1.c mytool1.h
gcc -c mytool1.c
mytool2.o:mytool2.c mytool2.h
gcc -c mytool2.c
有了这个Makefile文件,不过我们什么时候修改了源程序当中的什么文件,我们只要执行make命令,我们的编译器都只会去编译和我们修改的文件有关的文件,其它的文件它连理都不想去理的.
如何编写 makefile文件
在Makefile中以#开始的行都是注释行.Makefile中最重要的是描述文件的依赖关系的说明. 一般的格式是:
target: components
TABrule(TAB表示tab键)
第一行表示的是依赖关系.第二行是规则(命令行). 每一个命令行的开头都必须是一个tab键。
比如说我们上面的那个Makefile文件的第二行 main:main.o mytool1.o mytool2.o
表示我们的目标(target)main的依赖对象(components)是main.o mytool1.o mytool2.o 当倚赖的对象在目标修改后修改的话,就要去执行规则一行所指定的命令.就象我们的上面那个Makefile第三行所说的一样要执行 gcc -o main main.o mytool1.o mytool2.o
Makefile有几个非常有用的变量:分别是 $@、$*、$^、$< 。代表的意义分别是:
$@ -- 目标文件
$* -- 目标文件去掉后缀的部分
$^ -- 所有的依赖文件
$< -- 依赖目标中的第一个目标名字
如果我们使用上面几个变量,那么我们可以简化我们的Makefile文件为:
# 这是简化后的Makefile
main:main.o mytool1.o mytool2.o
gcc -o $@ $^
main.o:main.c mytool1.h mytool2.h
gcc -c $*.c
mytool1.o:mytool1.c mytool1.h
gcc -c $*.c
mytool2.o:mytool2.c mytool2.h
gcc -c $*.c