自动化构建工具Makefile/makefile

目录

一、makefile的背景

二、makefile的编写及其原理

            1.如何编写makefile

             2.依赖关系:

             3.依赖方法:

             4.makefile原理:

             5.项目清理:


一、makefile的背景

1. 也许长期使用Windows编程的程序员不知道这是什么东西,但是要在Linux下编译程序,你就不得不强制学习这个东西了。 

2.会不会写makefile,从一个侧面说明了一个人是否具备完成大型工程的能力,否则你就只能编写具有单个源文件的程序了。

 3. 一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,Windows中的编译环境已经替你做好了将这些

不同的源文件一起编译的准备工作,你只需要点击一下运行就可以了,但是Linux下可没有这么简单,makefile定义了一系列的规

则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作。

4.makefile的好处就是——自动化编译,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效

率。

二、makefile的编写及其原理

我们先写一个最简单的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/makefile_第1张图片

     上面的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

自动化构建工具Makefile/makefile_第2张图片

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

自动化构建工具Makefile/makefile_第3张图片

1>工程是需要被清理的,我们上面生成的test.i test.s test.o 和test这些临时文件都是要删除的,这时就要写clean了(代码第十行)。

2>像clean这种,没有被第一个目标文件直接或间接关联,那么它后面所定义的命令将不会被自动执行,不过,我们可以显示要

make执行。即命令——“make clean”,以此来清除所有的目标文件,以便重编译。

3>但是一般我们这种clean的目标文件,我们将它设置为伪目标,用 .PHONY (不用看时间,直接执行)修饰,伪目标的特性是,总是

被执行的。

 

 

 

你可能感兴趣的:(Linux)