【Linux】在Linux上写一个进度条小程序

news2025/1/18 7:03:52

👑作者主页:@安 度 因
🏠学习社区:安度因的学习社区
📖专栏链接:Linux

文章目录

  • 一、前言
  • 二、理解 '\r' 与 '\n'
  • 三、行缓冲
    • 1、提出问题
    • 2、认识行缓冲
    • 3、解答与拓展
    • 4、倒计时
  • 四、进度条
  • 五、结语

如果无聊的话,就来逛逛 我的博客栈 吧! 🌹

一、前言

在前三篇文章中,我们分别学习了 vim 、gcc 以及 make/makefile 。而在今天,我们将基于前三节课认识的基础上,并结合一些与回车换行、缓冲区有关的知识,在 Linux 上写下一个简易的进度条小程序。

成品展示

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jK64mBQM-1673533983679)(https://anduin.oss-cn-nanjing.aliyuncs.com/%E8%BF%9B%E5%BA%A6%E6%9D%A17.gif)]

今天的内容比较轻松,只需要了解两个知识点,这个小程序就很容易写出来了,让我们开始今天的学习。

二、理解 ‘\r’ 与 ‘\n’

C 语言中有很多字符,而字符大体分为两类:可显字符、控制字符。

控制字符不可显示,例如 \r 和 \n 就是控制字符。

而在我们平时打字时,一行写满了需要换行,但是新起一行有很多种,例如:

image-20230112173730693

这样虽然新起一行了,但是不是我们想要的结果。

我们通常新起一行是在第二行的最左端,但是对于这个结果其实有两个操作:

  1. 跳转到第二行
  2. 回到第二行的最左端

有了这个基本概念,再来谈 \r 和 \n 的作用:

  • \r :回车 - 回到文本行的开头
  • \n:换行 - 新起一行

所以,其实我们 平时泛指的换行实际上是 回车 + 换行

且在语言范畴下,例如 C 语言,换行就可以达到 回车 + 换行 的效果。在平常,这一操作还是两个步骤。

三、行缓冲

行缓冲这个概念认识。

1、提出问题

首先先了解一下两个库函数:

  • sleep : Linux 下的休眠函数,单位是秒。头文件为 #include <unistd.h>
  • fflush :刷新缓冲区

代码1:

#include <stdio.h>
int main()
{
    printf("hello xxx");
    sleep(3);
    return 0;
}

现象:

在这里插入图片描述

分析:

光标停留在文本行的开头,但是字符串没有被打印。

反而像是 sleep 函数先起作用,然后 printf 函数再从光标处开始打印。

打印完之后,shell 提示符紧跟着字符串后显示。

代码2:

#include <stdio.h>
int main()
{
    printf("hello xxx\n");
    sleep(3);
    return 0;
}

现象:

在这里插入图片描述

分析:

printf 打印的字符串先显示在终端上,光标位于字符串的下一行。

sleep 函数使程序休眠 3 秒后,shell 提示符从光标位置开始显示。

代码3:

#include <stdio.h>
int main()
{
    printf("hello xxx\r");
    sleep(3);
    return 0;
}

现象:

在这里插入图片描述

分析:

printf 打印的字符串没有显示到终端,光标一直停留在该打印字符串的一行

sleep 函数休眠三秒后,shell 提示符直接打印在了屏幕上。

并没有看到字符串。

观察上面的现象,我们提出几个问题:

  1. 代码 1 好像是先执行了 sleep ,在执行 printf ,是这样吗?
  2. 代码 2 加上了 ‘\n’ ,字符串一开始就显示了,为什么?
  3. 代码 3 好像什么都没打印,这是为什么?

在解答这些问题之后,我们先了解一下行缓冲。

2、认识行缓冲

在内存中预留了一块空间,用来缓冲输入或输出的数据,这个保留的空间被称为缓冲区。

我们之前或多或少都听说过缓冲区。

在代码 1 中,由于程序是按照数据执行的,所以必定是先执行 printf 。

但是数据没有显示,所以这时候,数据就一定被保存在某个位置,保存的位置就是缓冲区。

而要让数据显示,是需要刷新缓冲区的。

行缓冲是缓冲区刷新策略的一种,在行缓冲模式下,当输入和输出中遇到 ‘\n’ 换行时,就会刷新缓冲区

有了这个概念,我们继续分析问题。

3、解答与拓展

解答

问题1:代码 1 好像是先执行了 sleep ,在执行 printf ,是这样吗?

当然不是。

由于程序是按照顺序执行的,所以必定是先执行完 printf 在执行 sleep 。

而数据没有被显示出来的原因是:数据保存在缓冲区中,但是没有主动刷新,当程序退出后,保存在缓冲区中的数据被自动刷新出来了。

所以才会造成这种现象。

问题2:代码 2 加上了 \n ,字符串一开始就显示了,为什么?

这里由于是直接往显示器上打印,所以采用的刷新方式为行缓冲。

所以执行碰到 ‘\n’ 时,就会把在缓冲区中的(换行符之前)的内容全部刷新出来。

所以这段代码一开始就会有数据显示,然后再 sleep 休眠。

问题3:代码 3 好像什么都没打印,这是为什么?

之前说过 \r 是换行,所以当 printf 遇到 \r 时,就把光标移到开头。

sleep 睡眠后,当程序退出,shell 打印提示符时,就覆盖了字符串。

拓展

数据真的是临时保留在缓冲区里的吗?光标如何理解?

我们用一段代码来理解这两个问题:

#include <stdio.h>
#include <unistd.h>
int main()
{
    printf("hello xxx\r");
    fflush(stdout);
    sleep(3);
    return 0;
}

现象:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GCFKMagU-1673533983682)(https://anduin.oss-cn-nanjing.aliyuncs.com/%E8%BF%9B%E5%BA%A6%E6%9D%A14.gif)]

观察现象,我们发现当我们使用 fflush 主动刷新缓冲区后,数据就显示在了屏幕上;且因为 ‘\r’ 的原因,光标指向字符串开头;当打印 shell 提示符时,就直接从光标位置开始覆盖。

所以对于这两个问题,我们已经得到了答案:

  1. 数据被临时保存在于缓冲区中,通过刷新就可以显示
  2. 数据是从光标位置开始打印的。

一句话理解光标:光标和显示器匹配,光标在哪里,显示器打印的时候就从哪里开始打印

4、倒计时

基于对上面的理解,我们先实现一个简单的倒计时。

倒计时就是在屏幕上不断显示数字,每次在同一位置显示,并将之前的数据覆盖。

既然是每次要从头开始覆盖,那么就可以用 ‘\r’ 来实现每次回到行首,并且可以通过相应的格式化控制显示多位打印。

但是 ‘\r’ 不会主动刷新,所以要用 fflush 函数主动刷新缓冲区。

在每次刷新之后,使用 sleep 函数,间隔一定的时间。

由此,我们可以很轻松写出代码,例如写一个从 10 开始的倒计时:

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

int main()
{
    int i = 10;
    for (; i >= 0; i--) {
        // 位宽控制,\r 回到开头
        printf("%2d\r", i);
        fflush(stdout); // 主动刷新
        sleep(1); // 休眠
    }
    printf("\n"); // 换行,打印提示符
    return 0;
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vNYKYWQJ-1673533983682)(https://anduin.oss-cn-nanjing.aliyuncs.com/%E8%BF%9B%E5%BA%A6%E6%9D%A15.gif)]

四、进度条

好了,接下来进入正题,我们开始写 进度条

进度条样式

  • 主体样式为两个中括号包裹,中间 => 推进的方式呈现,比如:[======>]
  • 主体右侧中括号位置保持不变,中间元素不断推进,比如:[=> ]
  • 显示当前加载进度,用 [num%] 显示,num 随着进度条的不断推进而变化
  • 显示加载样式,可以利用一个旋转的字符,例如 [\] 的样式,顺时针不断旋转

大约呈现状态为:[========>] [15%] [\]

采用多文件

文件存放在 proc 目录中

  • proc.h :函数声明
  • proc.c :进度条逻辑
  • main.c :函数调用

makefile 准备

由于采用多文件,所以依赖关系可以写成依赖文件列表的样式:

image-20230112212907590

分块逻辑

  1. 进度条主体

预留进度条大小为 100 个 = ,外加 1 个 > ,加上保存 '\0' 的位置,用数组存储为 102 个单位。

进度条是一行中的,所以需要用到 '\r' ,每次都需要使用 fllush 主动刷新缓冲区。

每次刷新出数据之后,将 = 填充到数组中,并且显示 > 。在最后一次显示时,控制 > 不要显示。

然后休眠一小会。由于休眠用 sleep 函数太慢。所以可以用 usleep 函数休眠,usleep 函数的参数单位是微秒。

根据这个写出代码:

image-20230112220450323

在这里插入图片描述

  1. 百分比显示

%% 显示为一个 % ,而 %d 为数字,这步很简单,只要在 printf 语句中加上内容:

printf("[%-100s][%d%%]\r", bar, i);
  1. 旋转光标

光标旋转方向为顺时针旋转,那么旋转时就可以用数组保存。

旋转每次显示内容分别为 | / - \\\ 代表一个 \ ,因为和 \ 结合的会被解析为转义字符,将其保存到字符串中。

而由于字符串一共就四个字符,所以输出的时候需要控制输出位置。

代码:

const char* str = "|/-\\"; // 字符串
printf("[%-100s][%d%%][%c]\r", bar, i, str[i % 4]); // 输出语句

完整代码

proc.h

#pragma once 

#include <stdio.h>

extern void process();

proc.c

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

#define SIZE 102 // 数组大小
#define STYLE '='
#define FLAG '>'

void process()
{
    const char* str = "|/-\\";
    char bar[SIZE];
    memset(bar, '\0', sizeof(bar));
    int i = 0;
    while (i <= 100) {
      printf("[%-100s][%d%%][%c]\r", bar, i, str[i % 4]); // 格式控制
      fflush(stdout); // 刷新
      bar[i++] = STYLE; // 填充数据
      if (i != 100) {
          bar[i] = FLAG; // 如果不是最后一次则显示 >
      } 
      usleep(100000); // 休眠
    }
    printf("\n");
}

test.c

#include "proc.h"

int main()
{
    process();
    return 0;
}

进度条展示

在这里插入图片描述

五、结语

到这里,本篇博客就到此结束了。

今天的内容相对来说还是很简单的,我们的今天核心就是了解 ‘\r’ 和 ‘\n’ 的区别,并认识了行缓冲。只要了这两块知识,写起进度条就很轻松了。

大家感兴趣也可以下去试一试。

如果觉得 a n d u i n anduin anduin 写的不错的话,可以 点赞 + 收藏 + 评论 支持一下哦!我们下期见~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/158623.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

2023/1/12总结

今天学习了图的割点与桥的算法 图的割点以及桥 图的割点&#xff1a;割点是指在无向连通图中&#xff0c;某点和该点连接的边去掉以后图便不再连通 在上面的图片中&#xff08;上面是一个有向图&#xff0c;我们当作无向图即可&#xff09;我们知道当我们去掉A点之后&#xf…

进阶必看 | 6个让Revit建模起飞的习惯,高效就靠它

大家好&#xff0c;这里是建模助手。 相信各位都知道&#xff0c;建模助手一向以来都追求更高&#xff0c;更快&#xff0c;更强。但是有些问题&#xff0c;不是插件本身能解决的事情&#xff0c;而是项目本身的问题。 一般来说&#xff0c;当Revit项目模型大于150MB时&#…

Linux安装sonarqube(含各种错误处理)

目录 1.下载安装 2.错误处理 2.1.JDK版本不适配 2.2.can not run elasticsearch as root 1.下载安装 下载地址&#xff1a; Download | SonarQube | Sonar &#xff08;下载页面向下拉&#xff09;选择稳定版本下载。 解压后启动脚本在&#xff1a; bin/{对应操作系统}…

【dp】买卖股票的最佳时机系列题目

文章目录121. 买卖股票的最佳时机122. 买卖股票的最佳时机 II309. 最佳买卖股票时机含冷冻期123. 买卖股票的最佳时机 III188. 买卖股票的最佳时机 IV121. 买卖股票的最佳时机 本题的重点是&#xff1a;只能在前面某一天买入&#xff0c;后面某一天卖出。要不就是不买入&#x…

外贸业务员怎样能提高自己的工作能力?

关于外贸业务员提高自己的工作能力&#xff0c;米贸搜整理如下&#xff0c;希望可以帮助到你&#xff1a;1.树立一个好的目标&#xff0c;并坚定不移地朝着这个目标努力。这个问题&#xff0c;无论你是新手还是有经验的外贸业务员&#xff0c;相信每个外贸业务员都或多或少的思…

K_A11_004 基于STM32等单片机采集热敏传感参数串口与OLED0.96双显示

K_A11_004 基于STM32等单片机采集热敏传感参数串口与OLED0.96双显示一、资源说明二、基本参数参数引脚说明三、驱动说明IIC地址/采集通道选择/时序对应程序:四、部分代码说明1、接线说明1.1、STC89C52RC热敏传感模块1.2、STM32F103C8T6热敏传感模块五、基础知识学习与相关资料下…

NCS8823替代方案|CS5260Typec转VGA可替代NCS8823|低BOM成本替代NCS8823设计

NCS8823替代方案|CS5260Typec转VGA可替代NCS8823|低BOM成本替代NCS8823设计 NCS8823是一款低功耗、DisplayPort信号至VGA转换器,通过USB Type-C连接器。它是 适用于USB Type-C至VGA转换器&#xff0c;适配器、对接设备。此设备结合了基于USB Type-C的 DisplayPort接收器和VGA…

华为私有云平台FusionCompute搭建

一、FusionCompute架构 架构CNA作为虚拟化操作系统&#xff0c;VRM作为虚拟化管理平台正常主机都安装CNA&#xff0c;单独建立VRM集群作为管理集群&#xff0c;我测试环境就一台主机&#xff0c;所以CNA和VRM装在同一台主机上&#xff0c;并且用这台主机分配虚拟机进行测试。 …

前端基础(十二)_函数高级、全局变量和局部变量、 预解析(变量提升)、函数返回值

作用域 作用域指&#xff1a;变量或函数的有效使用范围&#xff0c;有全局作用域与局部作用域两种。 全局变量和局部变量 全局变量&#xff1a;直接在 script 标签下声明的变量&#xff0c;任何地方都能访问&#xff0c;任何地方都能对其值进行改变。 局部变量&#xff1a;函…

CAN总线的个人理解

部分内容可以参考&#xff1a;https://blog.csdn.net/xwwwj/article/details/105372234? CAN概念简介 CAN是Controller Area Network 的缩写 CAN协议经过ISO标准化后有两个标准&#xff1a;ISO11898标准和ISO11519-2标准。其中ISO11898是针对通信速率为125Kbps~1Mbps的高速通…

适合制造业的ERP推荐?使用ERP系统的好处有哪些?

对于制造型企业来说&#xff0c;除了涉及到产品的生产制造和原料采购&#xff0c;还需要管理库存、销售、财务等方方面面。制造业的ERP系统的使用&#xff0c;尤为重要。一个好的制造业的ERP系统在企业管理中起到至关重要的作用&#xff0c;针对制造业的ERP系统提供贴合行业特性…

用cmd命令窗口运行第一个java程序同时分享idea写的代码用cmd编译运行【建议收藏】

在上一篇文章https://blog.csdn.net/qq_52545155/article/details/128651296?spm1001.2014.3001.5502教大家安装了jdk版本&#xff0c;那么我们来编写一个java程序&#xff0c;通过cmd命令运行起来看看效果叭&#xff01;&#xff01;&#xff01; 一、基本代码准备 1、打开记…

超全的SQL注入姿势总结

目录 常见姿势 环境搭建 信息收集 报错注入 延时注入 布尔注入 堆叠注入 绕过方法 绕过引号 or and xor not绕过 绕过注释符 内联注释绕过 类型转换绕过 绕过 WAF绕过-应用层 常见姿势 环境搭建 use mysql; create table if not exists my_table( id int PRIMA…

HC小区管理系统安装记录一次群里小伙伴梓豪方式安装问题

记录一次群里小伙伴安装&#xff0c;供大家参考 问题排查 打开梓豪地址查看 Redis MySQL Nginx 是否启动成功&#xff0c;查看日志启动成功 MySQL正常 redis 没有报错 Nginx也正常 查看hc 是否启动成功&#xff0c;点击控制台 查看docker 发现8008 端口被占用了&#xff0c…

【异常】原来提示SocketTimeoutException:connect timed out还可能是外部因素导致

一、现象截图 一大早收到ELK的邮件提醒&#xff0c;让我来看看&#xff0c;又是哪个妖怪在作孽&#xff1f; 二、问题定位 2.1 SocketTimeoutException:connect timed out 经验告诉我&#xff0c;这个问题一般是第三方平台的问题&#xff0c;大部分原因是发起Http请求&…

惠普M329打印机更换副厂硒鼓后提示墨粉不足并无法打印

买了一个惠普M329打印机,打印效果不错,速度快,大量复印比较方便。因为最近打印和复印比较多,很快原装墨粉用完了。又买了一个副厂(带芯片)的硒鼓换上。不到一个月,又用光了,这次买了同一个副厂的硒鼓(不带芯片)。将原来的芯片(副厂的)拆下来,装在新硒鼓上。装到M3…

喜讯!华秋电子荣获千峰奖“2022年度产业互联网百强”,持续为电子产业增效降本!

12月28日&#xff0c;2022全球数字贸易大会专题活动“跨境产业互联网峰会暨2022年度千峰之夜”在武汉隆重举行&#xff0c;大会聚集了全国产业互联网、跨境电商平台、数字经济、资本市场等嘉宾&#xff0c;共同探讨产业互联网公司出海的机会和路径。 亿邦动力董事长郑敏介绍&am…

数字IC设计、验证、FPGA笔试必会 - Verilog经典习题 (三)奇偶校验

数字IC设计、验证、FPGA笔试必会 - Verilog经典习题 &#xff08;三&#xff09;奇偶校验 &#x1f508;声明&#xff1a; &#x1f603;博主主页&#xff1a;王_嘻嘻的CSDN博客 &#x1f9e8;未经作者允许&#xff0c;禁止转载 &#x1f511;系列专栏&#xff1a;牛客Verilog习…

房产管理系统平台架构安全需求分析

数图互通高校房产管理系统是基于公司自主研发的FMCenterV5.0平台&#xff0c;是针对中国高校房产的管理特点和管理要求&#xff0c;研发的一套标准产品&#xff1b;通过在中国100多所高校的成功实施和迭代&#xff0c;形成了一套成熟、完善、全生命周期的房屋资源管理解决方案。…

JavaAccessBridge(Java访问桥)详细介绍

Java Access Bridge 通过 Java 访问桥 API&#xff0c;您可以给使用Java应用程序编写的在windows系统上运行的图形化系统开发辅助技术应用程序。它包含本机方法&#xff0c;使您能够查看和操作有关 Java 应用程序中 GUI 元素的信息&#xff0c;这些信息将通过 Java 访问桥转发…