1、简介
Linux程序员必须学会使用GNU make 来构建和管理自己的软件工程。GNU 的make能够使整个软件工程的编译、链接只需要一个命令就可以完成。make在执行时,需要一个名为Makefile的文件。Makefile文件描述了整个工程的编译、链接等规则,其中包括:工程中有哪些源文件需要被编译以及如火如荼编译;需要创建哪些库文件以及如火如何创建这些库文件,如何最后产生我们想要得可执行文件。
2、Makefile中的词语
(1)规则:用于说明如火如荼生成一个或多个目标文件,规则格式如下:
targets : prerequisites command
目标 依赖 命令
例:main.o :main.c
gcc -c main.c
其中目标是:main.o 依赖是:main.c 命令是:gcc -c main.c
(2)目标:在Makefile中,规则的顺序是很重要的。因为,Makefile中只应该有一个最终目标,其他的目标都是最终目标的附属产物,所以,一定要让make知道你的最终目标是什么。一般来说,定义在Makefile中的目标可能有很多,但是第一条规则中的目标将被确立为最终的目标。
(3)make命令默认在当前目录下寻找名字为makefile或者Makefile的工程文件,当名字不为两者之一时,可以使用如下方法指定:
make -f 文件名
(4)伪目标
Makefile中把那些没有任何依赖只有执行动作的目标称为:“伪目标”(phony targets)
PHONY : clean
clean:
rm -f hello main.o func1.o func2.o
"PHONY"将“clean”目标声明为伪目标
(5)变量
在Makefile中存在系统默认的自动化变量
$︿:代表所有的依赖文件
$@:代表目标
$ㄑ:代表第一个依赖文件
例:hello:main.o func1.o func2.o
gcc main.o func1.o func2.o -o hello
等价于:
hello: main.o func1.o func2.o
gcc $︿ -o $@
(6)Makefile中“#”字符后的内容被视作注释
hello:hello.c
@gcc hello.c -o hello
@:表示取消回显
如果没有像windows下面的IDE,要想编译一个工程文件,需要怎样实现呢?例如:把windows下面的工程文件放到Linux下面,该怎样让它运行呢?下面我们先来了解两个概念:编译、链接。
当我们习惯了在windows下编程,我们写完程序后,只要点击一下按钮就可以执行,产生了可执行的文件。我们就会相当然的认为是这样:只要我们正确的写出程序然后点击按钮就可以了。其实,windows下的IDE为我们隐去了“编译”,“链接”。无论是C还是C++,首先要把源文件编译成中间代码文件,在Linux下就是“.o”后缀的文件,这个动作叫做编译。而链接就是把这些中间代码文件合成可执行文件。
Linux中,运行工程文件时,会用Makefile来完成编译连接的功能从而生成可执行文件。当我们将Makefile写好,然后运行make后,就会生成可执行文件。在Makefile中会定义一系列的规定,指定哪些文件先编译,哪些文件后编译等一系列的操作。假如在一个工程文件中,写好了正确的Makefile,感到好奇的是:为什么执行make后就会出现可执行文件,Make的工作原理是什么?
其实,当我们输入make后,make首先在当前目录下找到Makefile或者makefile文件,找到makefie后,会找到文件中的第一个目标文件ABC_1(自己假设的),并且会把此文件作为最终的目标文件。如果ABC_1不存在或者ABC_1后面依赖的ABC_2.o文件的修改时间要比ABC_1新,那么就会执行后面所定义的命令来生成ABC_1。但,如果ABC_1所依赖的ABC_2.o文件也不存在,那么make就会在此目录下找生成ABC_2.o所依赖的文件ABC_3.o,再根据生成规则生成ABC_2.o,就这样想一种递归的方式来寻找文件的依赖关系,直到找到最开始生成的目标文件。
下面是学习Makefile时写的两个程序:
源程序文件1:
Makefile文件:
运行的结果:
第一个程序的源文件就是一个简单的printf()函数,makefile中的hello就是最终要生成的目标文件,冒号后面是hello要依赖的文件,再后面就是生成hello所需要的命令。后面的"clean",和第一个目标文件没有联系,那么它定义的命令就不会自动的执行。因此,当要清除程序执行过程中产生的文件时就输入:make clean
源程序文件2:
Makefile文件:
运行的结果:
第二个程序中,有两个源文件,makefile中的Helloworld是最终要生成的目标文件,然后make会找到Helloworld的依赖文件main.o,此时main.o并不存在,make会继续寻找生成main.o的规则,生成main.o,然后make会找到Helloworld的依赖文件a.o,a.o也不存在,然后make找到生成a.o的规则,生成a.o,并最终生成了Helloworld