【Linux】LInux下第一个程序:进度条

1.回车换行符
大多数人可能觉得"回车换行"这一词指的是同一个东西。但其实回车是回车、换行是换行,这两个有本质区别。

我们日常使用的回车键合成了"回车"和"换行"功能,这就导致了大家觉得这是一个东西。

回车的符号为:\n
回车的功能是将光标重置到最开始位置

换行的符号为:\r
换行的功能是将光标下移一行

 

 

2.缓冲区问题
什么是缓冲区?

缓冲区是系统预留的内存区域,其作用是暂时存放输入或输出的数据。引入缓冲区主要是为了平衡高速的 CPU 与低速的 I/O 设备之间的速度差异,进而提升系统的整体性能。

刷新缓冲区?

通常情况下,数据会先被存储在缓冲区中,只有当缓冲区满、遇到特定的控制字符,或者程序运行结束时,才会将数据真正写入目标设备。不过,在某些特定场景下,可能需要手动刷新缓冲区,以确保数据能及时输出。

举例:

//代码1
#include<stdio.h>
#include<unistd.h>
int main()
{

printf("hello\n");
sleep(3);
}

//代码2
#include<stdio.h>
#include<unistd.h>
int main()
{
printf("hello");
sleep(3);
}

运行代码1:我们可以看到显示屏上马上显示出"hello",然后停滞3秒结束程序

运行代码2:我们看到是恰恰相反,先停滞3秒后打印出"hello",然后结束程序

这个就是缓冲区的问题:"\n"可以马上刷新缓冲区,所以代码1可以马上显示。而没有"\n",就无法马上刷新缓冲区,当整个程序结束后系统自动刷新,显示"hello"

那如何使其马上刷新?

fflush(stdout)

3.预备代码
先创建文件夹,并在文件夹中创建相应的.c .h文件

hyc@hcss-ecs-4ce7:~$ mkdir progress
hyc@hcss-ecs-4ce7:~$ cd progress
hyc@hcss-ecs-4ce7:~/progress$ touch p.c p.h main.c

编写代码,与makefile。进度条只会在一行打印,所以我们不能使用 \n,而是使用 \r,在同一行打印

 

 

但是当我们运行代码时,发现并没有显示结果。这是因为当前数据还在缓冲区没有刷新出来,我们使用ffulsh(stdout)使其马上刷新

 

这时我们可以看到结果:在同一位置上显示

 

4.进度条代码
4.1首先先看看我们想实现一个什么样子的进度条

4.2代码实现
我们一步一步来,先实现左侧部分

1.考虑使用字符数组来表示递增的进度条。先将字符数组初始化为 ‘\0’,通过%s打印时遇到 ‘\0’就会停止。

2.通过计数器计数,来计算输出多少符号

3.通过fflush(stdout)立马刷新缓冲区,让我们看到结果

4.通过sleep让我们看到其过程

5.最后为了避免命令行覆盖我们输出的内容,进行换行操作

 

最后添加一下左右中括号,修改一下细节问题:

打印进度条使用%-100s占位符

sleep有点太慢了,我们可以使用usleep

 

#include"p.h"
#include<unistd.h>
#include<string.h>

void progress_v1()
{
char arr[101];
memset(arr,'\0',sizeof(arr));
int num=0;
for(int i=0;i<=101;i++)
{
arr[num]='X';
printf("[%-100s]\r",arr);
fflush(stdout);
usleep(10000);
num++;
}
printf("\n");
}

 

接下来我们就可以来考虑一下百分比和旋转光标了

1.百分比直接去计算就好了,没什么好说的。

2.旋转光标:我们不要去想复杂了,动态的本质其实就是一帧一帧的静态图像,我们只需要写一个字符数组x,然后不断的打印字符,就可以实现旋转光标了。

 

#include<stdio.h>
#include<unistd.h>
#include<string.h>

void progress_v1()
{
char arr[101];
char x[]={'/','-','\\','\0'};
memset(arr,'\0',sizeof(arr));
int num=0;
for(int i=0;i<=101;i++)
{
arr[num]='X';
printf("[%-100s][%.2f][%c]\r",arr,num*1.0/sizeof(arr)*100,x[num%3]);
fflush(stdout);
usleep(10000);
num++;
}
printf("\n");
}

 

5.模拟真实环境
一个真正的进度条不可能就我们上面写的代码一样自顾自的打印。应该是根据实际情况,一边下载(或其他操作),一边打印进度条。

于是我们实现一个download函数模拟实时下载环境:

total:表示一共想要下载的量

speed:表示单次下载的速度

每下载一次就更新一次进度条

 

通过实时的下载量来计算进度条中需要打印的个数,以及百分比。

 

代码汇总:
//p.h

#include<stdio.h>
void progress_v1();
void progress_v2(double cur,double total);

//p.c

#include"p.h"
#include<unistd.h>
#include<string.h>

void progress_v1()
{
char arr[101];
char x[]={'/','-','\\','\0'};
memset(arr,'\0',sizeof(arr));
int num=0;
for(int i=0;i<=101;i++)
{
arr[num]='X';
printf("[%-100s][%.2f][%c]\r",arr,num*1.0/sizeof(arr)*100,x[num%3]);
fflush(stdout);
usleep(10000);
num++;
}
printf("\n");
}

void progress_v2(double cur,double total)
{
char arr[101];
char x[]={'/','-','\\','\0'};
memset(arr,'\0',sizeof(arr));

//get the number
int num=(int)(cur*100/total);
for(int i=0;i<num;i++)
{
arr[i]='X';
}

printf("[%-100s][%.2f][%c]\r",arr,cur*100/total,x[num%3]);
fflush(stdout);
}

//main.c

#include <stdio.h>
int main()
{
printf("Hello world\n");
return 0;
}

————————————————
版权声明:本文为CSDN博主「Yuzuriha Inori」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/huangyuchi/article/details/148436299

阅读剩余
THE END