【Linux取经路】进度条小程序

06-01 531阅读

【Linux取经路】进度条小程序

文章目录

  • 一、预备知识
    • 1.1 回车换行
    • 1.2 缓冲区
    • 二、倒计时
      • 2.1 注意事项
      • 三、进度条
        • 3.1 源代码
        • 3.2 代码分析
        • 3.2 实际使用场景

          一、预备知识

          1.1 回车换行

          一般意义上的回车换行是两个独立的独立的动作,而C语言中的\n则同时完成了回车和换行的工作。回车是将光标移动到当前行的做开始(最左侧),换行是将光标水平方向保持不变,竖直方向向下平移一行。C语言中可以通过转义字符\r实现回车。

          【Linux取经路】进度条小程序

          我们电脑键盘上的EBTER按键则是同时实现了回车和换行的功能,当按下ENTER键,光标会去到下一行的最开始的位置。

          1.2 缓冲区

          📖先看一个现象

          #include 
           int main()                                     
          {
               printf("Hello Linux!\n");      
                                                        
               sleep(2);
               return 0;
          }
          

          这段代码很简单,现在屏幕上打印出Hello Linux!,接着调用sleep函数让程序休眠两秒。接下来,我们对上面的代码稍作修改,去掉\n再来试试。

          #include 
           int main()                                     
          {
               printf("Hello Linux!");      
                                                        
               sleep(2);
               return 0;
          }
          

          【Linux取经路】进度条小程序

          通过动图可以看到,在去掉/n后对代码编译运行,先是休眠了两秒,接着才在屏幕上打印出Hello Linux!,并且因为没有\n,所以打印完后没有换行,导致bash命令行就紧跟在打印结果的后面。

          📖现象分析

          很多小伙伴会根据上面的现象猜测,这段代码先执行了sleep休眠,再去执行printf打印,这样的猜测是错误的,因为任何一个C程序,都是严格按照代码的顺序去执行。既然这样的话,先执行printf,再执行sleep,那在休眠的两秒期间,printf的打印结果在哪里呢?由于最终Hello Linux!还是出现在我们的屏幕上,所以在这两秒期间,Hello Linux一定是被保存起来了,其实就是保存在缓冲区中。缓冲区就是C语言维护的一段内存。默认当程序结束的时候才会将缓冲区中的内容刷新出来。

          📖如何强制刷新缓冲区

          任何一个C程序运行的时候都会默认帮我们打开以下三个流:

          • stdin - - - - 标准输入流(键盘)
          • stdout - - - - 标准输出流(显示器)
          • stderr - - - - 标准错误(显示器)

            【Linux取经路】进度条小程序

            Linux下一切介文件,这三个流都是FILE*的指针,所以任何一个C程序运行的时候,操作系统会帮我们打开以上三个文件。今天我们只需要关心stdout标准输出流即可。我们可以通过fflush函数来刷新缓冲区。

            【Linux取经路】进度条小程序

            #include 
             int main()                                     
            {
                 printf("Hello Linux!");      
                 fflush(stdout);//刷新缓冲区                                      
                 sleep(2);
                 return 0;
            }
            

            【Linux取经路】进度条小程序

            通过运行结果可以看出,这一次虽然在打印的时候也没有加\n,但取先把Hello Linux!打印出来,然后再休眠两秒。

            小Tips:通过上面的分析我们可以得出,刷新缓冲区主要有以下几种方法:

            • \n可以刷新缓冲区。
            • 程序结束也会刷新缓冲区。
            • fflush(stdout)可以手动刷新缓冲区。

              二、倒计时

              有了上面的知识储备,我们先来实现一个简单的倒计时练练手。

              📖源代码

              #include "processBar.h"
              #include 
              int main()
              {
              	int cnt = 10;
              	while(cnt >= 0)
              	{
              		printf("%-2d\r",cnt);
              		fflush(stdout);
              		sleep(1);
              		cnt--;
              	}
              	
              	printf("\n");                            
              	return 0;
              }
              

              📖效果演示

              【Linux取经路】进度条小程序

              2.1 注意事项

              📖回车、刷新缓冲区

              由于倒计时,是用新数字去覆盖老数字,因此每打印一个数字后不能用\n进行换行,否则就会像下面这样:

              【Linux取经路】进度条小程序

              这里的正确做法是,每打印一个数字后紧跟着打印一个\r回车,让光标回到这一行最开始的位置,这样新打印的数字就会去覆盖掉老的数字。但是\r不会去刷新缓冲区,因此在每打印完一个数字后,需要调用fflush(stdout)来刷新缓冲区。

              📖格式化控制

              这里我们需要知道,往显示器上打印整型10,本质上是打印了字符1和字符0,由于这两个字符是挨在一起的,我们看起来就像是整型10。因此打印10,会占用两个字符,而打印0~9只需要一个字符,所以\r回车之后去覆盖写,只会覆盖一个字符,对第二个字符0始终没有影响,因此我们需要用%-2d来控制,每次打印两个位宽的字符,-表示将这两个字符左对齐。如果不进行格式化控制,打印出来的结果将是下面这样:

              【Linux取经路】进度条小程序

              三、进度条

              3.1 源代码

              📖processBar.h

              #pragma once
              #include 
              #define NUM 102
              #define STYLE '='                            
              #define TOP 100
              #define BODY '>'
              extern void processbar();
              

              📖processBar.c

              #include "processBar.h"
              #include 
              #include 
              const char* lable = "|/-\\";//旋转提示
              void processbar()
              {
                  char bar[NUM];
                  memset(bar, '\0', sizeof(bar));
                  int len = strlen(lable);
                  int cnt = 0;
                  while(cnt 
                      printf("[%-100s][%d%%][%c]\r", bar, cnt, lable[cnt%len]);
                      fflush(stdout);
                      bar[cnt++] = STYLE;
                      if(cnt ,第二次打印出来的就是=>,对标1%。当到进度到达100%的时候,我们希望打印出来的进度条右边没有>,因为100%对应的是最后一次打印,也就是当cnt == 100的时候,此时我们希望打印出100个=即可,这意味着,当执行这次打印时,数组下标为99的位置存储的是一个=并且下标为100的位置是\0,前者简单,当cnt == 99的时候字符串追加的时候会把其设置成=,要满足后者,我们就要加一个判断条件当cnt ,否则不能修改bar[cnt]。 
              

              3.2 实际使用场景

              上面的processBar.c中为了演示进度条的原理,在里面写了一个while循环来模拟,但实际上的进度条并不是这样用的。以下载东西为例,作为一个进度条,它本身并不知道下载了多少,它只会提供一个接口,在下载东西的时候,调用这个接口,然后将已经下载好的比率作为参数传给进度条模块,它会根据比率打印出对应的进度条样式。

              📖版本一

              //processBar.h
              #pragma once
              #include 
              #define NUM 102
              #define STYLE '='
              #define TOP 100
              #define BODY '>'
              extern void processbar(int ret);
              
              //processBar.c
              #include "processBar.h"
              #include 
              #include 
              const char* lable = "|/-\\";
              //V2版本
              char bar[NUM] = {'\0'};//定义在全局避免每一次函数调用都会重现创建                      
              void processbar(int ret)
              {
              	if(ret  100)//合理性判断
              	{
              		return;
              	}
              	if(ret == 0)//当比率为0的时候将数组全置为'\0'
              	{
              		memset(bar, '\0', sizeof(bar));
              	}
              	int len = strlen(lable);
              	printf("[%-100s][%d%%][%c]\r", bar, ret, lable[ret%len]);
              	fflush(stdout);
              	bar[ret++] = STYLE;
              	if(ret  
              
              //main.c
              int main()
              {  
                  int total = 1000;//假设总共要下载1000个G  
                  int cur = 0;//当前下载的  
                  while(cur      
                      processbar(cur * 100 / total);                   
                      usleep(50000);//模拟下载花费时间                 
                      cur += 10;//循环下载了一部分,更新进度           
                  }     
                 return 0;   
              }
              '\0'};
              void processbar(int ret)
              {
              	if(ret 
              		return;
              	}
              	if(ret == 0)//当比率为0的时候将数组全置为'\0'
              	{
              		memset(bar, '\0', sizeof(bar));
              	}
              	int len = strlen(lable);
              	printf("["LIGHT_BLUE"%-100s"NONE"]""[%d%%][%c]\r", bar, ret, lable[ret%len]); 
              	fflush(stdout);                        
              	bar[ret++] = STYLE;
              	if(ret 
免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们。

目录[+]

取消
微信二维码
微信二维码
支付宝二维码