目录
一、makefile的背景
二、makefile的编写及其原理
1.如何编写makefile
2.依赖关系:
3.依赖方法:
4.makefile原理:
5.项目清理:
1. 也许长期使用Windows编程的程序员不知道这是什么东西,但是要在Linux下编译程序,你就不得不强制学习这个东西了。
2.会不会写makefile,从一个侧面说明了一个人是否具备完成大型工程的能力,否则你就只能编写具有单个源文件的程序了。
3. 一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,Windows中的编译环境已经替你做好了将这些
不同的源文件一起编译的准备工作,你只需要点击一下运行就可以了,但是Linux下可没有这么简单,makefile定义了一系列的规
则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作。
4.makefile的好处就是——自动化编译,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效
率。
我们先写一个最简单的hello world:
1 #include
2 int main()
3 {
4 printf("hello world\n");
5 return 0;
6 }
1.如何编写makefile
接下来编写makefile文件:
1 test:test.c
2 > gcc test.c -o test
它的意思就是说:我要靠test.c源文件生成一个test文件,我要怎么生成呢?就靠第二行的这个gcc命令生成。
写好之后,只需要在命令行输入一个make,编译器就会自动编译,然后运行test即可。当然如果只有一个简单的源文件可以直接用在命令行用gcc命令,这里只是为了演示makefile的用法。
上面的makefile只用了一条gcc命令,但其实我们知道一个源文件要通过预处理、编译、汇编、链接四个步骤才形成一个可执行文件,我们写一个详细的makefile来看看makefile到底是如何运行的:
1 test:test.o
2 > gcc test.o -o test
3 test.o:test.s
4 > gcc -c test.s -o test.o
5 test.s:test.i
6 > gcc -S test.i -o test.s
7 test.i:test.c
8 > gcc -E test.c -o test.i
2.依赖关系:
test , 它依赖 test.o
test.o , 它依赖 test.s
test.s , 它依赖 test.i
test.i , 它依赖 test.c
3.依赖方法:
gcc test.* -option test.* ,就是与之对应的依赖关系
4.makefile原理:
我们输入make命令之后:
1. make会在当前目录下找名字叫“Makefile”或“makefile”的文件。
2. 如果找到,它会找文件中的第一个目标文件,在上面的例子中,他会找到“test”这个文件,并把这个文件作为最终的目标文件。
3. 如果test文件不存在,或是test所依赖的后面的test.o文件的文件修改时间要比hello这个文件新,他就会执行后面所定义的命令
来生成hello这个文件。
4. 如果test所依赖的test.o文件不存在,那么make会在当前文件中找目标为test.o文件的依赖性,如果找到则再根据那一个规则生
成hello.o文件。(有点像一个堆栈的过程)
5. 当然,代码编写正常的情况下3和4是不会发生的,于是make会生成 test.o 文件,然后再用 test.o 文件声明make的终极任务,
也就是执行文件hello了。
6. 这就是整个make的依赖性,make会一层又一层地去找文件的依赖关系,直到最终编译出第一个目标文件。
7. 在找寻的过程中,如果出现错误,比如最后被依赖的文件找不到,那么make就会直接退出,并报错,而对于所定义的命令的
错误,或是编译不成功,make根本不理。
8. make只管文件的依赖性,即,如果在我找了依赖关系之后,冒号后面的文件不在,那么它就不工作了。
$^ 表示该依赖方法对应的依赖关系对应的文件列表
$@表示该依赖方法对应的依赖关系对应的目标文件
例:test:test.c
(一个tab键)gcc -o $@ $^
其中$@对应的是test $^对应的是test.c
5.项目清理:
1 test:test.o
2 > gcc test.o -o test
3 test.o:test.s
4 > gcc -c test.s -o test.o
5 test.s:test.i
6 > gcc -S test.i -o test.s
7 test.i:test.c
8 > gcc -E test.c -o test.i
9
10 .PHONY:clean
11 clean:
12 > rm -f test.i tset s test.o test
1>工程是需要被清理的,我们上面生成的test.i test.s test.o 和test这些临时文件都是要删除的,这时就要写clean了(代码第十行)。
2>像clean这种,没有被第一个目标文件直接或间接关联,那么它后面所定义的命令将不会被自动执行,不过,我们可以显示要
make执行。即命令——“make clean”,以此来清除所有的目标文件,以便重编译。
3>但是一般我们这种clean的目标文件,我们将它设置为伪目标,用 .PHONY (不用看时间,直接执行)修饰,伪目标的特性是,总是
被执行的。