当前位置: 首页 > news >正文

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

前言:

        在前面的文章中我们学习了LInux的基础指令

      【Linux】初见,基础指令-CSDN博客【Linux】初见,基础指令(续)-CSDN博客

        学习了vim编辑器【Linux】vim编辑器_linux vim insert-CSDN博客

        学习了gcc/g++【Linux】编译器gcc/g++及其库的详细介绍-CSDN博客

        以及make/makefile【Linux】自动化构建-Make/Makefile-CSDN博客

有了以上知识的铺垫,我们终于可以开始在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 numberint 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;
}                                                                                                                                         
http://www.xdnf.cn/news/922159.html

相关文章:

  • 十一、【ESP32开发全栈指南: TCP通信服务端】
  • 1-3 Linux-虚拟机(2025.6.7学习篇- mac版本)
  • Sentry 接口返回 Status Code 429 Too Many Requests
  • 【优选算法】C++滑动窗口
  • 在ubuntu等linux系统上申请https证书
  • Redis内存淘汰策略
  • redis集群
  • [最全总结]城市灾害应急管理系统
  • Linux虚拟化技术:从KVM到容器的轻量化革命
  • Nodejs工程化实践:构建高性能前后端交互系统
  • sqlsugar WhereIF条件的大于等于和等于查出来的坑
  • WSL文件如何上传到GitHub
  • python版若依框架开发:后端开发规范
  • 快捷键的记录
  • UOS无法安装deb软件包
  • [论文阅读] 人工智能 | 搜索增强LLMs的用户偏好与性能分析
  • AcWing--数据结构1
  • stm32—ADC和DAC
  • 《JavaAI:稳定、高效、跨平台的AI编程工具优势解析》
  • Linux下的fuser用法简析
  • 文件(保存)通讯录
  • 长跑赛接力赛模式
  • C++ -- 多态
  • 《高等数学》(同济大学·第7版)第二章第五节“函数微分“
  • SpringBoot+Mysql校园跑腿服务平台系统源码
  • Doris 与 Elasticsearch:谁更适合你的数据分析需求?
  • 游戏常用运行库合集 | GRLPackage 游戏运行库!
  • LILIKOI FBG腹腔镜抓握力传感器的技术解析与应用前景
  • 调试器基本原理
  • LeetCode 08.06 面试题 汉诺塔 (Java)