Linux项目自动化构建工具-make/Makefile
make 是一条指令,makefile 是一个文件,两个搭配使用,完成项目自动化构建。
makefile :
makefile 是一个文件。它是一个工程文件的编译规则,描述了整个工程的编译链接等规则。
写好的 makefile 文件可以使用一行命令来完成 “自动化编译” ,从而完成对工程的编译,极大提高效率。
make :
make 是一个命令工具,用来一个解释 makefile 中文件中的指令。
当已经编写好 makefile 文件后,只需要使用 make ,就可以执行 makefile 中的内容。
make/Makefile实现
makefile 文件需要创建在当前工程的目录下,makefile 文件的名称的首字母不区分大小写,我们先写一段代码
在控制台输入make即可完成编译
make/Makefile实现原理
我们知道 test 是目标文件, test.c 是原始文件,然后 test.c 经过 gcc test.c -o test 生成 test 文件。
那么他们两个之间的关系就是
test 依赖 test.c 生成,所以 test.c 是 test 的依赖文件 。它们之间的关系被称为 依赖关系 。
test.c 生成 test 需要通过 gcc test.c -o test 指令,这条指令就是 依赖方法 。
我们通过一个 makefile 理解一下:
在这个makefile文件中,有四组依赖关系和依赖方法,当使用 make 调用 makefile 文件中内容时,便开始执行 makefile 中的内容:
test 依赖于 test.o ,
test.o 依赖于 test.s ,
test.s 依赖于 test.i ,
test.i 依赖于 test.c ,test.c 存在,这时开始执行依赖方法
可以看到,依赖方法对应的文件都产生了,所以依赖关系和依赖方法必须同时具备并正确,缺一不可 。
clean指令
工程是需要被清理的,像clean这种,没有被第一个目标文件直接或间接关联,那么它后面所定义的命令将不会被自动执行, 不过,我们可以显示要make执行。即命令——“make clean”,以此来清除所有的目标文件,以便重编译。
这里说一下 .PHONY伪目标
.PHONY 修饰的对象是伪目标,伪目标的特性是:总是被执行的。
.PHONY 修饰的一定能被反复执行,但是能被反复执行的不一定被 .PHONY 修饰。
可以看到makeclean被.PHONY修饰后每次都可执行。
一般对于编译来说,是不加 .PHONY 修饰的。
因为编译是十分耗时间,特别是当工程量很大的时候,编译一两小时都不为过。所以防止对未修改的程序反复编译 ,一般编译时不加修饰。
但是 清理clean 是可以多次执行的,因为删除不太浪费时间,且可以反复清理,确认是否清理完毕。并且为了肯定清理可以被多次执行,所以通常用 .PHONY 修饰。
make判断是否能编译的条件
上面我们测试 make 时,发现当编译过一次后,继续使用 make 就无法继续编译了。但是 clean 是可以不加修饰反复执行的。 make 是如何确定是否要编译?
对于程序来说,时间有两条线。第一条是源代码时间的一条线,第二条是形成的可执行程序的时间的一条线 。而它们之间的次序,是先有源代码,再有可执行程序。
所以只要可执行程序的最近修改时间比源文件的修改时间晚,就认为当前可执行程序是最新的,为了减少时间和其他开销,于是不执行编译;反之执行编译 。
我们使用stat指令查看文件的时间变动
我们可以看到,Test比Test.c晚,这时候再make就不起作用了。
总结:
对于依赖关系而言,: 左边为目标文件,: 右边为依赖文件
依赖方法前需要有一个 tab ,为固定格式
: 右边可以有多个依赖文件 ,: 右边通常被称为依赖文件列表
对于 : 右边,目标文件对应的依赖文件列表可以为空 (例如 clean)
makefile 默认执行第一组的依赖关系和依赖方法,对于第一组可以直接使用 make 执行,后面则需要 make + 目标文件
Linux 进度条小程序
行缓冲区
#include <stdio.h>
int main()
{
printf("hello Makefile!\n");
sleep(3);
return 0;
}
此代码,是先输出字符串hello Makefile然后休眠3秒之后结束运行。
#include <stdio.h>
int main()
{
printf("hello Makefile!");
sleep(3);
return 0;
}
此代码,删除了字符串后面的’\n’,运行是先休眠3秒,然后打印字符串hello Makefile之后结束运行
显示器对应的是行刷新,即当缓冲区当中遇到’\n’或是缓冲区被写满才会被打印出来,而在第二份代码当中并没有’\n’,所以字符串hello Makefile先被写到缓冲区当中去了,然后休眠3秒后,直到程序运行结束时才将hello Makefile打印到显示器当中。
\r: 回车,使光标回到本行行首。
\n: 换行,使光标下移一格。
这里我们可以用fflush()函数刷新缓冲区,把缓冲区当中的数据刷新到显示器中。
#include <stdio.h>
int main()
{
printf("hello Makefile!");
fflush(stdout);
sleep(3);
return 0;
}
下面就是简单的进度条实现