目录
背景
- 会不会写makefile,从一个侧面说明了一个人是否具备完成大型工程的能力
- 一个工程中的源文件不计数,其按类型、功能、模块分别放在若干个目录中,makefile定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作
- makefile带来的好处就是——“自动化编译”,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高了软件开发的效率。
- make是一个命令工具,是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如:Delphi的make,Visual C++的nmake,Linux下GNU的make。可见,makefile都成为了一种在工程方面的编译方法。
- make是一条命令,makefile是一个文件,两个搭配使用,完成项目自动化构建。
创建一个Makefile文件
创建一个Makefile文件需要哪些内容:a. 依赖关系 b. 依赖方法
上图中hello为目标文件,一般我们还会写一个clean的伪目标用于清理目标文件,伪目标用“.PHONY”修饰。(什么是伪目标文件,下面会介绍)
创建好Makefile,我们只需要输入指令:make 目标文件,即可执行相关依赖方法,如:make hello; make clean.当然也可以只输入make 这样系统会只执行Makefile文件里从上往下的第一个目标文件(包括伪目标文件)。
接下来,我们介绍一下什么是伪目标!
伪目标文件特点:总是被执行!什么意思?看下图:
简单创建一个多文件项目:
创建多个可执行文件:
.PHONY:all
all:test hello
test:test.c
gcc -o $@ $^ //$@ 表示依赖关系前部分(test),$^ 表示依赖关系后部分(test.c)
hello:hello.c
gcc -o $@ $^
.PHONY:clean
clean:
rm -f test hello
Linux下的一个小程序-进度条
写进度条之前,我们先介绍2个概念:a) 行缓冲概念 b) 回车,换行区别
行缓冲区概念
下面代码各现象:
#include <stdio.h>
#include<unistd.h>//sleep的头文件
int main()
{
printf("hello Makefile!\n");
sleep(3);
return 0;
}
现象:先打印语句,再休眠3秒
int main()
{
printf("hello Makefile!");
sleep(3);
return 0;
}
现象:先休眠3秒,再打印语句
int main()
{
printf("hello Makefile!");
fflush(stdout);
sleep(3);
return 0;
}
现象:先打印语句,再休眠3秒
上述现象不同的原因是什么?难道是执行语句顺序不同?不是!main函数中执行语句一定是从上往下执行。第二段代码printf语句先于sleep语句执行,只不过信息没有立马显示出来。
C语言给我们提供了输出缓冲区(一段内存空间),打印语句显示送入缓冲区内,然后才被刷新出来。
第一段代码解释:显示器设备,一般的刷新策略是行刷新,碰到‘\n’就把‘\n’之前所有内容显示出来。所有先打印printf语句再休眠3秒。
第三段代码解释:fflush(stdout)把缓冲区的代码刷新出来。所以先打印printf语句再休眠3秒。
回车,换行区别
回车 = 换行?答案是否定的!
换行是直接换到下一行,即从光标位置直接往下,不会回到行首; 回车是回到当前光标所在行的行首。
换行符号:\n 回车符号:\r 下面简单证明一下。
先用 \n:
接下来看 \r :
原因:前面缓冲区概念得知,printf会把语句先打印到缓冲区,\n 有刷新缓冲区能力,而 \r 不会刷新,所以每次语句都没打印到缓冲区里,而且由于 \r 会回到行首,所以即便到了最后也没有把内容刷新出来,若想刷新出来,我们需要fflush一下:
这样就会出现类似屏幕闪烁的效果,就像进度条原理一样。