0


Linux第一个小程序——进度条

回车和换行

回车\r:**'r' 回车**,回到当前行的行首,而不会换到下一行,如果接着输出的话,本行以前的内容会被逐一覆盖。

换行\n:**'n' 换行**,换到当前位置的下一行,而不会回到行首。(满足回车/换行两种功能)

缓冲区

  1 #include<stdio.h>
  2 #include<unistd.h>//sleep
  3 int main()
  4 {
  5   printf("hello linux,hello word");                                                                          
  6   sleep(3);
  7   return 0;
  8 }
  1 #include<stdio.h>
  2 #include<unistd.h>//sleep
  3 int main()
  4 {
  5   printf("hello linux,hello word\n");                                                                          
  6   sleep(3);
  7   return 0;
  8 }

屏幕录制 2024-01-27 194734

执行上面两段代码(它们的区别是是否包含\n)

我们清晰的发现:

  • 带有\n的就直接出现在了显示器(屏幕)上
  • 没有带\n的程序结束之后才回显示在终端上

结论:包括\n在内的之前全部字符串全部刷新到我们的显示器上(行刷新)

刷新的方式:

  • ** \n**
  • 缓冲区满了
  • **程序结束 **
  • 强制刷新fflush fflush(stdout)
  • 头文件#include<unistd.h> //sleep所需要的头文件包含【Linux系统编程】Linux的系统库中的unistd.h头文件的作用-CSDN博客
  • fflush - C++ Reference (cplusplus.com)
  • stdout - C++ Reference (cplusplus.com)
//输出流--显示器/终端
extern FILE * stdout;
extern FILE * stderr;
//输入流--键盘
extern FILE * stdin;
//开机默认打开三个流(输入/输出流)
  //修改之后
  1 #include<stdio.h>
  2 #include<unistd.h>//sleep
  3 int main()
  4 {
  5   printf("hello linux,hello word");
  6   fflush(stdout);                                                                                            
  7   sleep(3);                                                                                
  8   return 0;                                                                                
  9 } 

为什么是行刷新??缓冲区的存在又是为了什么?立即刷新?

  • 行刷新既考虑到了用户的读取(用户一般是一行一行读取),又考虑到了效率问题。
  • 缓冲区的存在就是为了提高效率。
  • 立即刷星就是我们在终端/显示器从键盘输入文字字符,立即出现在用户面前。
  • 应用思考:Linux上输入密码不显示的原因。

设计倒计时

  1 #include<stdio.h>
  2 #include<unistd.h>
  3 int main()
  4 {
  5   int cnt=10;
  6   while(cnt>=0)
  7   {
  8     printf("倒计时:%2d\r",cnt);//2d  
  9     fflush(stdout);//刷新数据
 10     cnt--;
 11     sleep(1);//休眠
 12   }
 13   printf("\n");                                                                                              
 14   return 0;
 15 }
  • 原理:同一个位置用不同的数字做刷新,所以是动态的变化的 。
  • \r回车,光标回到最开始覆盖前面数据,历史数据在缓冲区已经被fflush刷新到终端上了
  • 2d 是两个字符占位符,我们输入123,输入的是字符1 字符2 字符3 所以10需要两个占位符。
  • 显示器是字符设备。
  • sleep函数引用**#include<unistd.h>**

**最终呈现效果: **

屏幕录制 2024-01-28 122350

进度条(多文件操作)

  • Processbar.h 包含头文件&函数声明
  • Processbar.c函数实现
  • Main.c函数调用/测试
  • Makefile自动化项目构建
  • Makefile里面不需要包含头文件Processbar.h,因为头文件在当前目录下且Processbar.c和Main.c都包含了Processbar.h在自动化构建编译时,回展开头文件,所以不需要!!

** 测试一下:**

Version1:进度条

#include"Processbar.h"
  2 //void test()
  3 //{
  4 //  printf("this is a test\n");
  5 //  printf("this is a test\n");
  6 //  printf("this is a test\n");
  7 //  printf("this is a test\n");
  8 //  printf("this is a test\n");
  9 //  printf("this is a test\n");
 10 //}     
                                                                                                     
 11 #define length 101 //包括\0
 12 #define DataType '#'
 13 const char*MoveType = "|/-\\";
 14 void ProcBar()
 15 {
 16   char Bar[length];//缓存空间
 17   memset(Bar,'\0',sizeof(Bar));
 18   int cnt=0;
 19   while(cnt<=100)
 20   {
 21     printf("[%-100s][%3d%%][%c]\r",Bar,cnt,MoveType[cnt%strlen(MoveType)]);
 22     fflush(stdout);
 23     Bar[cnt++]=DataType;
 24     usleep(20000);//0.02s=20000微秒
 25   }
 26   printf("\n");
 27 }

屏幕录制 2024-01-28 151404

注释:

  • memset设置为了遇到\0停止打印,如果不清空全是随机值打印。
  • 单位换算
  • printf一般是右对齐,加上符号-左对齐
  • %% 打印出 %
  • \ 是反斜杠\的字面常量
  • 占位符100% 3%
  • 取余是为了防止数组越界!!
  • 强制刷新fflush(stdout)
  • **设计-/\ **进度条一直在加载/下载,但是因为网速等原因,卡住不动,但是此刻用户又想看到一个进度条在下载的动态效果


Version2:应用场景+进度条

  • 下载文件filesize
  • 当前下载量currentsize
  • 带宽/网速bandwidth
  • 根据网速/宽带的不同,使其下载文件的速度也有差别,进度条的快慢也有差别
  • 单位换算1兆=1024*1024字节
  • 占比currentsize/filesize*100.0
  • 强制类型转化
  • 浮点数打印lf
  • 回调函数&函数指针复习回顾

Processbar.h

  1 #pragma once //防止头文件被重复包含
  2 #include<stdio.h>
  3 #include<unistd.h>
  4 #include<string.h>
  5 //extern 表示此函数/变量在其他文件别处定义
  6 //extern void test();
  7 typedef void(*callbact_t)(double,double);
  8 void ProcBar(double filesize,double currentsize);
  9 // void Download(double filesize,callbact_t cb); 

Processbar.c

    #include"Processbar.h"
 34 #define length 101 //包括\0
 35 #define DataType '#'
 36                                                                                                              
 37 const char*MoveType = "|/-\\";
 38 void ProcBar(double filesize,double currentsize)
 39 {
 40   char Bar[length];//缓存空间
 41   memset(Bar,'\0',sizeof(Bar));
 42   int cnt=0;
 43   double rate=currentsize*100.0/filesize;//占比
 44   int count=(int)rate;
 45   while(cnt <= count)
 46   {
 47    // printf("[%-100s][%.1lf%%][%c]\r",Bar,rate,MoveType[cnt%strlen(MoveType)]);
 48    // fflush(stdout);
 49     Bar[cnt++]=DataType;
 50    // usleep(20000);//0.02s=20000微秒
 51   }
 52     printf("[%-100s][%.1lf%%][%c]\r",Bar,rate,MoveType[cnt%strlen(MoveType)]);
 53     fflush(stdout);
 54 }

Main.c

  1 #include"Processbar.h"
  2 //设置一个场景
  3 //double filesize=100*1024*1024*1.0;//总的文件大小
  4 //double currentsize=0.0;//目前下载的文件数
  5 double bandwidth=1024*1024*1.0;
  6 void Download(double filesize,callbact_t cb)                                                                 
  7 {
  8   double currentsize=0.0;
  9     printf("Download begin,currentsize:%lf\n",currentsize);
 10   while(currentsize <= filesize)
 11   {
 12     cb(filesize,currentsize);
 13     usleep(20000);
 14     currentsize+=bandwidth;
 15   }
 16      printf("\nDownload done,currentsize:%lf\n",currentsize);
 17 }
 18 int main()
 19 {
 20   Download(100*1024*1024*1.0,ProcBar);
 21  //ProcBar();
 22  //test();
 23   return 0;
 24 }

Makefile


  1 Processbar:Processbar.c Main.c
  2   @gcc -o $@ $^
  3 .PHONY:clean
  4 clean:
  5   @rm -f Processbar 

屏幕录制 2024-01-29 140123


Version3:升级彩色进度条

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

int main()
{
    int k=0;
     int j=0;
     int color[]={4,3,2,5,6};
     char bar[102];
     memset(bar,0,sizeof(bar));
     const char* str="|/-\\";
      for(;k<=100;++k)
     {
         printf("\033[3%dm[%-100s]\033[0m\033[33m[%d%%]\033[0m[%c]\r",                  
             color[j], bar, k, str[k % 4]);
        fflush(stdout);
        bar[k]='*';
        //每20%变色一次
        if(k%20==0)
        {
            ++j;
        }
        usleep(7000);
     }
     printf("\n");
     return 0;
 }

🙂感谢大家的阅读,若有错误和不足,欢迎指正

标签: linux 运维 服务器

本文转载自: https://blog.csdn.net/m0_74841364/article/details/135870212
版权归原作者 唐唐思 所有, 如有侵权,请联系我们删除。

“Linux第一个小程序——进度条”的评论:

还没有评论