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

小程序 0

目录

回车和换行

缓冲区

设计倒计时

进度条(多文件操作)

Version1:进度条

Version2:应用场景+进度条

Processbar.h

Processbar.c

Main.c

Makefile 

Version3:升级彩色进度条


回车和换行

回车/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; }

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

也许您对下面的内容还感兴趣: