💛不要有太大压力🧡
💛生活不是选择而是热爱🧡
Linux下第一个小程序:进度条
两个背景知识
缓冲区
举一个例子
我们用<unistd.h>
中的sleep()
函数来休眠,单位(秒)
以上程序:先输出hello,world!然后等待三秒程序结束
但是如果去掉换行符,如下:
程序的运行结果:先等待三秒,然后输出hello,world!
这是为什么呢?
只有一种可能:printf早就执行完了, 只是内容没有输出到显示器
C语言给我们提供了输出缓冲区,输出的内容先流入缓冲区中,而缓冲区按照一定的刷新策略来刷新,刷新后才会显示到 屏幕
所谓的输出缓冲区,其实就是C语言提供的一段内存空间
其实,printf是输出到stdout(标准输出流)的,对应的就是显示器。
对于显示器设备,一般的刷新策略是:碰到’\n’就把’\n’之前的所有字符显示出来(包括’\n’)
如果想立即刷新:fflush(FILE* steam)
函数,参数是文件流
C语言有三个标准流:stdin``stdout``stderr
如图:在stdio.h中声明
所以,当代码改成这样的时候就可以先打印出内容然后再执行sleep函数了
回车和换行是一个概念?
no!
回车:回到当前行的最开始
换行:光标垂直向下移动一行
所以,当前行输入之后,来到下一行的最开始其实是:回车+换行
但是其实键盘上的回车帮我们干了这两件事
分开来看,回车:\r
换行:\n
而其实在语言中的’\n’是被当作回车+换行的,所以在语言中有’\n’就自动来到下一行的最开始了
看当前代码,输出结果是
当前的值是5
当前的值是4
当前的值是3
当前的值是2
当前的值是1
但是如果把’\n’换成’\r’,如下图
结果是什么都不会输出
因为只有’\r’,没有’\n’,每一次数据输入到缓冲区之后没有被刷新('\n’刷新缓冲区)。程序运行结束缓冲区会自动刷新,此时缓冲区的内容是:当前的值是:5\r当前的值是:5\r当前的值是:4\r当前的值是:3\r当前的值是:2\r当前的值是:1\r
,这些内容会一起刷新到显示器,由于光标马上回到最开头,所以什么都不会输出
那么如何强制刷新呢?
用fflush
:强制把缓冲区的内容刷新到缓冲区
此时的输出结果就是
当前值是5 (清屏)
当前值是4 (清屏)
当前值是3 (清屏)
当前值是2 (清屏)
当前值是1
也就有了倒计时的效果
为什么呢?
- 因为每次把
当前的值是%d\r
放入缓冲区,然后强制刷新缓冲区就把该内容打印到了屏幕,并且因为sleep,屏幕保持睡眠,不会马上把光标回到开头,sleep结束,光标回到开头,又从头打印下一次的值 - 因为每一次打印的字符串长度是一样的,每一次打印完光标回到当前行的最开始,然后强制刷新缓冲区到屏幕,下一次,相同的值进行覆盖就看不到变化,只有最后的数字不同,所以只有后面的值发生变化
注意:‘\n’和’\r’只有到了显示器才会有效果,在缓冲区只是个字符
进图条原理
进度条代码
#include<stdio.h>
#include<string.h>
#include<unistd.h>
#define NUM 101
int main()
{
char ch[NUM]={0};
memset(ch,0,NUM);
int cur=0;
char* label="|/-\\";//四个状态不停旋转,表示正在加载
while(cur<=100)
{
//%100s:打印的字符串始终占100个字符,符号是靠左(因为C语言默认是靠右占位)
printf("[%-100s][%d%%](%c)\r",ch,cur,label[cur%4]);//模4是为了不超过数组索引(循环四个字符)
ch[cur++]='=';//每一次多一个#
fflush(stdout);//每一次进行刷新
usleep(50000);// 单位是微秒(1s=10^6us)
}
printf("\n");
return 0;
}
效果
其他玩法:蹦迪版进度条
这里需要用到Linux系统终端的文字颜色,是文本模式下的系统显示功能,与具体语言无关
转义序列以控制字符’ESC’开头。该字符的ASCII码十进制表示为27,十六进制表示为0x1B,八进制表示为033。多数转义序列超过两个字符,故通常以’ESC’和左括号’['开头。该起始序列称为控制序列引导符(CSI,Control Sequence Intro),通常由\033[
或 \e[
代替
不同颜色由\033[xm
中的数字x决定 (固定格式末尾有个m)
常见的文字颜色:
30:黑
31:红
32:绿
33:黄
34:蓝色
35:紫色
36:深绿
37:白色
举个例子
如果想让输出helloworld为红色
printf("\033[31m")
:如果不把文字设置为原来的终端颜色,之后就不会变了
而颜色的参数(30)所以我们可以利用一个变量来动态改变
#include<stdio.h>
#include<string.h>
#include<unistd.h>
#define NUM 101
int main()
{
char ch[NUM]={0};
memset(ch,0,NUM);
int cur=0;
int x = 1;
char* label="|/-\\";//四个状态不停旋转,表示正在加载
while(cur<=100)
{
x=x%7+1;//红到白色7中颜色
if(cur==100)
{
x=7;//最后一次恢复白色
}
printf("\033[3%dm[%-100s][%d%%](%c)\r",x,ch,cur,label[cur%4]);//模4是为了不超过数组索引(循环四个字符)
ch[cur++]='=';//每一次多一个#
x++;
fflush(stdout);//每一次进行刷新
usleep(50000);
}
printf("\n");
return 0;
}
效果
✨感谢阅读~ ✨
❤️码字不易,给个赞吧~❤️