(创建于 2018/2/7 上午7:43:55)
MakeFile 构建工作,只需通过make一句命令就可以构建一个可执行程序,Maven ANT Gradle都是构建工具
在很多C/C++开源项目中,configure文件用来检查系统配置生成配置文件,这些配置Makefile文件用来生成我们需要的动态库文件
为什么要写Makefile文件?
1.项目非常庞大时,让构建过程自动化简单
2.当依赖文件比目标文件有更新,会重建目标文件
(gcc命令前必须是tab键,解析的时候根据tab键来判断这是个命令,所有tab键后边的内容都会被当作命令对待,所以如果写注释,要注意)
Makefile文件相当于一系列的gcc 命令的组合
gcc -c xxx.c 生成.o文件(相当于javac xxx生成class文件)
gcc -o xxx.o 链接生成可执行文件
如下图,这个文件就是Makefile文件,文件名就是Makefile,固定的
#include<stdio.h>
int divide(int a,int b) {
printf("a/b=%d\n",a/b);
}
#include<stdio.h>
int multi(int a, int b) {
printf("axb=%d\n", a * b);
}
#include<stdio.h>
int jian(int a, int b) {
printf("a-b=%d\n", a - b);
}
#include<stdio.h>
int plus(int a, int b) {
printf("a+b=%d\n", a +b);
}
int plus(int a, int b);
int jian(int a, int b);
int multi(int a, int b);
int divide(int a, int b);
void main() {
plus(20,10);
jian(20, 10);
multi(20, 19);
divide(20, 10);
}
myapp:main.o plus.o minus.o multi.o divi.o
gcc main.o plus.o minus.o multi.o divi.o -o myapp
#目标:main.o
#依赖:main.c
#命令:gcc -c main.c
main.o:main.c
gcc -c main.c
plus.o:plus.c
gcc -c plus.c
minus.o:minus.c
gcc -c minus.c
multi.o:multi.c
gcc -c multi.c
divi.o:divi.c
gcc -c divi.c
clean:
rm -f *.o
rm -f myapp
执行make命令得到如下,生成了很多.o临时文件,如何在生成之后删除这些文件?
file:///C:/Users/renzhenming/Documents/My%20Knowledge/temp/4d3d1996-855f-40c3-9375-1f9fc118cb27/128/index_files/951526656.png
删除临时文件
myapp:main.o plus.o minus.o multi.o divi.o
gcc main.o plus.o minus.o multi.o divi.o -o myapp
#目标:main.o
#依赖:main.c
#命令:gcc -c main.c
main.o:main.c
gcc -c main.c
plus.o:plus.c
gcc -c plus.c
minus.o:minus.c
gcc -c minus.c
multi.o:multi.c
gcc -c multi.c
divi.o:divi.c
gcc -c divi.c
//设置一个命令clean,删除相关的东西(可见clean这个命令是自己写在Makefile文件中的,并非关键字,可以随意写,只要我们调用的时候写对就可以了,比如这里写clean,执行makefile文件的时候写make clean,如果写remove ,那命令自然就是make remove了)
clean:
rm -f *.o
rm -f myapp
执行make命令(只输入一个make),就可以执行最终的目标也就是写在最上边的生成myapp的命令,如果你不要执行最终目标,而是像执行其中的一个目标,比如,我要执行生成main.o文件,那么就可以执行这个命令,make main.c 你要执行哪个就写哪个文件的名字,前边加make既可
可以看到如main.o:main.c 意思时main.o依赖于main.c生成,回车后执行gcc -c main.c生成main.o文件
最终生成的可执行文件myapp依赖于main.o plus.o munus.o multi.o divi.o
:后边的都是依赖文件,一个可执行文件可能依赖于很多文件,然而并不是每次生成可执行文件的时候所有依赖文件都会被修改,
所以再次生成的时候只需要重新构建被修改的依赖文件即可,Makefile具备检查文件是否更新的功能,再次编译生成时,它会去重构已修改的文件
当所有文件都没有被修改时,执行make结果如下截图
file:///C:/Users/renzhenming/Documents/My%20Knowledge/temp/4d3d1996-855f-40c3-9375-1f9fc118cb27/128/index_files/780668562.png
当我们修改了其中一个c文件后执行结果:(main.c被更新,导致main.o被重构,main.o重构后,myapp也需要被重构)
file:///C:/Users/renzhenming/Documents/My%20Knowledge/temp/4d3d1996-855f-40c3-9375-1f9fc118cb27/128/index_files/780778109.png
myapp:main.o plus.o minus.o multi.o divi.o
2 gcc main.o plus.o minus.o multi.o divi.o -o myapp
3
4 #目卤辏?ain.o
5 #?5拢潞main.c
6 #??拢潞gcc -c main.c
7 main.o:main.c
8 gcc -c main.c
9 plus.o:plus.c
10 gcc -c plus.c
11 minus.o:minus.c
12 gcc -c minus.c
13 multi.o:multi.c
14 gcc -c multi.c
15 divi.o:divi.c
16 gcc -c divi.c
17
18 .PHONY:clean //为了防止存在和命令有相同命名的文件产生冲突,这里设置伪命令
19 clean:
20 rm -f *.o
21 rm -f myapp
添加命令之前执行结果
file:///C:/Users/renzhenming/Documents/My%20Knowledge/temp/4d3d1996-855f-40c3-9375-1f9fc118cb27/128/index_files/781545156.png
添加伪命令之后执行结果
file:///C:/Users/renzhenming/Documents/My%20Knowledge/temp/4d3d1996-855f-40c3-9375-1f9fc118cb27/128/index_files/781575437.png
上边那种是初级写法,接下来我们升级一下,这样就有点类似Android mk文件了
OBJECTS=main.o plus.o minus.o divi.o multi.o
myapp:$(OBJECTS)
gcc $^ -o $@ //#自动化变量 $^表示所有依赖,$@表示目标
%.o:%.c //#通配符 #gcc -c main.c -o main.o
gcc -c $^ -o $@
.PHONY:clean
clean:
rm -f *.o
rm -f myapp
接下来第三阶段写法
1 SOURCES=$(wildcard *.c) //#所有.c源文件
2 OBJECTS=$(patsubst %.c,%.o,$(SOURCES)) //#把SOURCES中的所有.c后缀,替换成.o后缀
3
4 myapp:$(OBJECTS)
6 gcc $^ -o $@
8 %.o:%.c
10 gcc -c $^ -o $@
16 .PHONY:clean
17 clean:
18 rm -f *.o
19 rm -f myapp