本文最近学习makefile,为方便以后回忆对相关知识点进行了梳理。
Makefile 的优点在于可以实现“自动化编译”。首先需要把源文件编译成中间代码文件,windows下为.obj文件,unix下是.o文件,即objectfile。这个动作叫做编译。然后再把大量的objectfile合成执行文件,这个动作叫做链接。在实际情况中,由于源文件太多,编译生成的中间目标文件太多,而在链接时需要明显地指出中间目标文件名,这样对于编译很不方便,所以需要对中间目标文件打包,windows下这种包叫做“库文件”即.lib文件,在unix下为ArchiveFile即.a文件。总而言之,源文件首先会生成中间目标文件,然后通过中间目标文件生存执行文件。编译时编译器只检测程序语法,函数、变量是否被声明。如果函数未被声明,编译则会给出警告,但是可生存目标文件。而链接程序时,链接器会在所有的目标文件中找寻函数的实现,若找不到则会报链接错误。
首先编写main.c,add.h,add.c三个文件
main.c
#include "add.h"
#include <iostream>
using namespace std;
int main()
{
cout<<"2+3="<<add(2,3)<<endl;
return 0;
}
add.h
#ifndef ADD_H
#define ADD_H
int add(int a,int b);
#endif
add.c
#include "add.h"
int add(int a,int b)
{
return (a+b);
}
Makefilede的规则为:
target...:prerequisites...
command
...
...
target是一个目标文件,此处可以是objectfile,也可以是执行文件,或者还可以是标签。
prerequisites指target所需要的文件或者目标。
command是make需要执行的命令。
当prerequisites中如果有意个以上的文件比target文件要新的话,从哦满满的所定义的命令就会被执行。
然后编写Makefile文件
test1: main.o add.o
g++ -o test1 main.o add.o
main.o:main.c add.h
g++ -c main.c add.h
add.o:add.c add.h
g++ -c add.c add.h
clean:
rm test1 main.o add.o
在Makefile中定义依赖关系后后续要以一个Tab建作为开头。编写完之后保存,输入make会在文件目录下看到test1,main.o,add.o三个文件,其中test1为执行文件,其余两个为中间目标文件,最后执行./test1即可看到结果2+3=5
从上面的Makefile可以发现.o文件重复次数角度,当在工程中需要加入新的.o文件时,需要在多出修改,如果忘掉一处修改,则会导致编译失败。所以我们采用定义变量的方式改写Makefile:
objects=main.o add.o
test1:$(objects)
g++ -o test1 $(objects)
main.o:main.c add.h
g++ -c main.c add.h
add.o:add.c add.h
g++ -c add.c add.h
clean:
rm test1 main.o add.o
每个Makefile中都需要写一个清空目标文件的规则,这样做的好处是不仅便于重新编译,而且可以保持文件的清洁。这里我们采用:
.PHONY:clean
clean:
-rm test1 $(objects)
其中.PHONY表示clean是一个伪目标,而rm前面加一个小减号意为当某个文件出现问题时,不中断然后继续做后面的事情。