本文的内容为Linux下的工程组织管理方案make和makefile以及利用make和makefile实现一个进度条。
make是一个命令,makefile是一个文件
通过依赖关系和依赖方法来实现自动化编译工作,自动化构建工程和清理
如果源文件没有发生更改,则不能再进行make
通过ldd可以看到,使用的时默认链接方法,动态链接
make默认是从上往下扫描,默认执行的是第一个依赖关系,make clean 执行Makefile中的clean依赖关系
依赖关系解释的是为什么
语法:目标文件:依赖文件列表(依赖文件1 依赖文件2 ...)
依赖方法解释的是怎么样
语法:【Tab】命令
make和makefile在源文件不修改的情况下默认只能生成一次,其原因是比较可执行程序的最近修改时间和源文件的最近修改时间,只要可执行程序的最近修改时间比所有源文件的最近修改时间新,则说明当前可执行程序是最新的。注意其比较的时间是Modify时间
查看文件时间命令:
stat 文件名
Access:文件访问时间
Modify:内容修改时间
Change:属性修改时间
修改文件的属性后,Change时间发生改变
文件的内容发生改变,Change时间也会发生改变,原因是改变了文件的内容,文件的大小也会发生改变,文件大小属于文件的属性
每一次访问文件,Access时间不一定会发生变化,因为访问的频率比较高,为了减少修改时间的频率,系统设定为不是每次访问都更新时间
1)为了解除make对于只能创建一个工程的限制,可以通过.PHONY:文件名,将文件名修饰为目标文件,成为一个伪目标,总是被执行
mybin成为伪目标后,可以一直make,一般将clean设置为伪目标而创建工程一般不设置为伪文件,因为由于个别文件的更新导致整个项目的更新,只需要编译发生更改的文件即可,可以大大提高编译的效率
2)可以使用$@和@^代替目标文件和源文件
例如:
3)可以使用变量名替换内容,在使用时,将需要替换的地方改为$(变量名)即可。
例如:
4)关于make/makefile语法推导问题
1)预备知识:
1.缓冲区问题
C对IO函数提供了一个缓冲区,是一块内存空间,强制刷新函数fflush( 流 )
在c语言中,会默认打开三个流stdin、stdout、stderr
sleep(int n)函数,睡眠n秒,usleep(int n)函数:睡眠n微秒
2.回车与换行
回车和换行在严格意义上不是同一个概念,换行是光标到下一行的同一个位置,而换行是回到同一行的开始位置,C语言中\r的作用是回车。
2)实现
用一个字符数组模拟进度条,在每一次响应时将进度调进行刷新,填充进度条信息,并将新的进度条打印到屏幕上,使用usleep每0.01秒刷新进度条。
#include"processbar.h" #define size 101 #define label '#' const char*Label="|/-\\"; void process(){ char buffer[size]; memset(buffer,' ',sizeof(buffer)); buffer[size-1]='\0'; int cnt=0; while(cnt<101){ printf("[%s]%3d/100 %c\r",buffer,cnt,Label[cnt%4]); buffer[cnt++]=label; fflush(stdout); usleep(50000); } printf("\n"); }
导入模拟下载过程+进度条
#include"processbar.h" #define size 101 #define label '#' const char*Label="|/-\\"; const unsigned int FILESIZE = 1024*1024*1024; char buffer[size]= {0}; unsigned int flag=0; void process(double rate){ static int flag = 0; int cnt = (int)rate; memset(buffer,label,cnt); printf("\033[0m\033[45;36m[%-100s]%5.2f%% %c\r\033[0m",buffer,rate,Label[flag%4]); fflush(stdout); flag++; } void download(){ unsigned int total = 0; srand(time(NULL)^1023); while(totalFILESIZE) total = FILESIZE; process((1.0*total)/FILESIZE*100); usleep(10000); } printf("\n"); memset(buffer,'\0',sizeof(size)); }