linux—makefile编译规则(一)

make命令和makefile文件的结合提供了一个在项目管理领域十分强大的工具。

makefile文件由一组依赖关系和规则构成。


makefile的规则:先写目标的名称,然后紧跟这一个冒号,接着是空格或者制表符tab,最后是用空格或者制表符tab隔开的文件列表(这些文件用于创建生成目标文件)。

target ... : prerequisites ...
command
...
...


目标文件可以是中间目标文件,也可以是可执行文件,还可以是标签、伪目标。
command也就是make需要执行的命令(任意的shell命令)。
prerequisites中如果有一个以上的文件比target文件要新的话,command所定义的命令就会被执行。这就是Makefile的规则。也就是Makefile中最核心的内容。


例如:编写文件Makefile1
  myapp: main.o 2.o 3.o
gcc -o myapp main.o 2.o 3.o
main.o:main.c a.h
gcc -c main.c
2.o:2.c a.h b.h
gcc -c 2.c
3.o:3.c b.h c.h
gcc -c 3.c


它表示目标myapp依赖于main.o 2.o 3.o,而main.o依赖于main.c和a.h 等等。
$make -f  Makefile1  
/*-f 因为makefile文件并未使用常见的默认文件名makefile或者Makefile */
make: *** No rule to make target 'main.c',needed by 'main.o'. Stop.


解释:make命令假设在makefile文件中的第一个目标文件myapp 是想要创建的目标文件,然后它会检查其他的依赖关系,并确定需要一个main.c的文件。由于并未创建该文件,而且makefile文件中也并未说明如何创建,所以make命令报告一个错误。


下面我们创建这些源文件,由于我们对程序的结果没有兴趣,所以文件的内容非常简单,头文件都为空。


$touch a.h
$touch b.h
$touch c.h


[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:14px;">/*main.c*/  
  2. #include <stdio.h>  
  3. #include <stdlib.h>  
  4. #include <string.h>  
  5. #include <math.h>  
  6.   
  7. extern void function_1();  
  8. extern void function_2();  
  9.   
  10. int main()  
  11. {  
  12.     function_1();  
  13.     function_2();  
  14.     return 0;  
  15. }</span>  

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:14px;">/*2.c*/  
  2. #include <stdio.h>  
  3. #include <stdlib.h>  
  4. #include <string.h>  
  5. #include <math.h>  
  6. #include "a.h"  
  7. #include "b.h"  
  8.   
  9. void function_1(){  
  10.     printf("hello world\n");  
  11. }</span>  

[cpp]  view plain  copy
  在CODE上查看代码片 派生到我的代码片
  1. <span style="font-size:14px;">/*3.c*/  
  2. #include <stdio.h>  
  3. #include <stdlib.h>  
  4. #include <string.h>  
  5. #include <math.h>  
  6. #include "b.h"  
  7. #include "c.h"  
  8. void function_2(){  
  9.     printf("hello linux\n");  
  10. }</span><span style="font-size: 18px;">  
  11. </span>  

/*再次执行make命令*/
$make -f Makefile1
gcc -c main.c
gcc -c 2.c
gcc -c 3.c
gcc -o myapp main.o 2.o 3.o


/*生成可执行文件myapp ,我们运行以下这个程序*/
$./myapp
hello world
hello linux



make命令处理makefile 文件中定义的依赖关系,确定需要创建的文件以及创建顺序。虽然把如何创建目标myapp 列在最前面,但make命令能够自行判断出创建文件的正确顺序。他调用你在规则部分给出的command来创建相应的文件,同时会在执行时在屏幕上将命令显示出来。
现在我们来测试一下 在文件b.h改变时,makefile文件能否正确处理这一情况:
$touch b.h
$make -f Makefile1
gcc -c 2.c
gcc -c 3.c
gcc -o myapp main.o 2.o 3.o


make命令读取makefile文件,确定需要重建myapp所需的最少命令,并以正确的顺序执行它们。
$rm 2.o
$make -f Makefile1
gcc -c 2.c
gcc -o myapp main.o 2.o 3.o



make命令再次正确地确定出 需要采取的动作。

详细易懂的Linux makefile教程 一、概述 什么是makefile?或许很多Winodws的程序员都不知道这个东西,因为那些Windows的IDE都为你做了这个工作,但我觉得要作一个好的和professional的程序员,makefile还是要懂。这就好像现在有这么多的HTML的编辑器,但如果你想成为一个专业人士,你还是要了解HTML的标识的含义。特别在Unix下的软件编译,你就不能不自己写makefile了,会不会写makefile,从一个侧面说明了一个人是否具备完成大型工程的能力。 因为,makefile关系到了整个工程的编译规则。一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,makefile定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作,因为makefile就像一个Shell脚本一样,其中也可以执行操作系统的命令。 makefile带来的好处就是——“自动化编译”,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效率。make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如:Delphi的make,Visual C++的nmake,Linux下GNU的make。可见,makefile都成为了一种在工程方面的编译方法。 现在讲述如何写makefile的文章比较少,这是我想写这篇文章的原因。当然,不同产商的make各不相同,也有不同的语法,但其本质都是在“文件依赖性”上做文章,这里,我仅对GNU的make进行讲述,我的环境是RedHat Linux 8.0,make的版本是3.80。必竟,这个make是应用最为广泛的,也是用得最多的。而且其还是最遵循于IEEE 1003.2-1992 标准的(POSIX.2)。 在这篇文档中,将以C/C++的源码作为我们基础,所以必然涉及一些关于C/C++的编译的知识,相关于这方面的内容,还请各位查看相关的编译器的文档。这里所默认的编译器是UNIX下的GCC和CC。 二、关于程序的编译和链接 —————————— 在此,我想多说关于程序编译的一些规范和方法,一般来说,无论是C、C++、还是pas,首先要把源文件编译成中间代码文件,在Windows下也就是 .obj 文件,UNIX下是 .o 文件,即 Object File,这个动作叫做编译(compile)。然后再把大量的Object File合成执行文件,这个动作叫作链接(link)。 ...
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值