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

OSE3.【Linux】练习:编写进度条及pv命令项目中的进度条函数

目录

1.前置知识

回车和换行

sleep函数

缓冲区的简单理解

fflush函数

usleep函数

2.预备实验:倒计时程序

3.进度条项目文件结构一览

预备

编写进度条

版本1

版本2:加上中括号

版本3:添加百分比

版本4:添加旋转光标动画

版本5:其他风格

gitee仓库代码

4.模拟进度条打印函数的调用

模拟带参的进度条函数

使用回调函数调用

知识回顾

download函数模拟下载过程

5.从pv命令项目看进度条显示函数的调用

复制文件时打印进度条显示进度

生成2GB的测试文件

使用pv命令复制测试文件

部分源代码分析


1.前置知识

回车和换行

回车和换行的复习:参见77.【C语言】文件操作(3)文章

注意:回车( \r )不等于换行( \n )

sleep函数

函数声明: unsigned int sleep(unsigned int seconds);

和Windows下的Sleep不同,Linux下的sleep函数首字母小写

man指令查询C语言sleep的内容:

 man 3 sleep #注意查的是3号手册(库调用)

 列出以下要点

1.头文件:<unistd.h>

2.参数:指定睡眠的秒数(unsigned int)

缓冲区的简单理解

以下代码在Linux的运行结果是什么?

#include <stdio.h>
#include <unistd.h>
int main()
{printf("test string");sleep(3);//睡眠3sreturn 0;
}

 答案:

会发现并没有先打印字符串,而是先睡眠3s

但程序又是从上到下依次执行的,因此字符串一定是先被保存起来了,这个被保存的地方称为缓冲区,是由C语言所维护的一段内存空间

如果想立刻看到字符串的打印结果可以使用fflush(stdout)强制刷新,因为显示器模式是行刷新,如果没有fflush函数,必须有\n才能刷新

fflush函数

(来自https://legacy.cplusplus.com/reference/cstdio/fflush/?kw=fflush) 

flush v.刷新 则fflush 将标准输出流(stdout)的输出缓冲区中的数据强制写入到目标输出设备(通常是终端或文件)中

注:IO流的有关内容参见75.【C语言】文件操作(2)文章

#include <stdio.h>
#include <unistd.h>
int main()
{printf("test string");fflush(stdout);sleep(3);//睡眠3sreturn 0;
}

运行结果:程序立刻打印了字符串,之后2s结束

usleep函数

函数声明:

int usleep(useconds_t usec);

简单来说,和sleep函数不同的地方在于参数的单位是微秒

2.预备实验:倒计时程序

#include <stdio.h>
#include <unistd.h>
int main()
{for (int i=10;i>=0;i--){printf("%-2d\r",i);//右对齐两格fflush(stdout);sleep(1);        }return 0;
}

注意: 1. \r是让光标回到本行开始处,为了覆盖前面的数字 2.必须设置位宽,右对齐两格,为了让数字正确显示

编译命令:

gcc检查比较严格,必须要带c99标准,否则for循环报错

gcc test.c -o test -std=c99

运行结果:

3.进度条项目文件结构一览

|- processbar.c  #含processbar函数的定义

|- processbar.h #含processbar函数的声明以及其他头文件

|- main.c #调用processbar函数

|- makefile #构建进度条项目

预备

//processbar.h
#pragma once
#include <stdio.h>
#include <unistd.h>
#include <string.h>
extern void processbar();//表明该函数在其他c文件中
#makefile
processbar:processbar.c main.c #没有写头文件,编译器会自动找gcc -o $@ $^
.PHONY:clean
clean:rm -f processbar
//main.c
#include <processbar.h>
int main()
{processbar();return 0;
}

编写进度条

稍微修改下预备实验写到 processbar.c中即可,几点要求:

1.将要输出的百分数存入buffer数组,每次循环打印buffer数组即可

2.buffer数组初始化时要清零

3.每次输出百分数后都回车(\r)

4.随着时间的流逝,尾插到buffer的进度条的部分会越来越多

5.0%打印0个#,100%打印100个#,而且有效字符串的结尾必须有\0,再预留一个安全字节因此buffer数组的大小为102个字节

版本1

#define CAPACITY 101
#define STYLE '#' //进度条的样式
void processbar()
{char buffer[CAPACITY];memset(buffer,'\0',sizeof(buffer));int cnt=0;while (cnt<=100){printf("%s\r",buffer);//必须带上\rfflush(stdout);buffer[cnt++]=STYLE;usleep(8000);}printf("\n");
}

运行结果:

版本2:加上中括号

改下printf即可,注意左对齐

printf("[%-100s]\r",buffer);

版本3:添加百分比

改下printf即可,%%解释为%

printf("[%-100s] %3d%%\r",buffer,cnt);

运行结果:

版本4:添加旋转光标动画

光标按顺时针旋转,动画依次为 \ | / -

可以定义一个字符数组animation来存储动画,依次播放即可

char* animation="\\|/-";//注意\\解释为一个反斜杠
字符\|/-
下标0123

 播放代码

//cnt%4余数在0~3之间,对应不同的动画字符
printf("[%-100s] %3d%% %c\r",buffer,cnt,animation[cnt%4]);

也可以将4代换成animation_len,s其值为strlen(animation) 

运行结果:

版本5:其他风格

例如实现这个的风格  [======>                       ] ??%

 ======>分为两个部分,一个是由=组成的进度条的主要部分,一个是单个的>

定义STYLE和HEAD:

#define STYLE '='
#define HEAD '>'

改变对buffer尾插的方式:

buffer[cnt++]=STYLE;
buffer[cnt]=HEAD;

稍微修改下printf

printf("[%-101s] %3d%%\r",buffer,cnt);//buffer的1最后一个安全字节留给>

运行结果:

或者到100%时删除>

 在循环结束时添加以下代码即可:

cnt--;//cnt变成100
buffer[cnt]=STYLE;
buffer[cnt+1]='\0';
printf("[%-101s] %3d%%\r",buffer,cnt);
fflush(stdout);

运行结果:

gitee仓库代码

https://gitee.com/zhangz6/c-code/tree/master/linux%E8%BF%9B%E5%BA%A6%E6%9D%A1%E4%BB%A3%E7%A0%81

4.模拟进度条打印函数的调用

使用yum或apt安装软件包时会打印进度条,简单来讲是将软件下载的百分比传递给processbar函数

而项目中传给进度条显示函数的参数往往较为复杂,参见下方对pv命令项目的部分源代码的解析 

模拟带参的进度条函数

那就不能使用内部循环,rate的大小是由外部告知的

char buffer[CAPACITY];//全局
void processbar(int rate)
{memset(buffer,'\0',CAPACITY);buffer[rate++]=STYLE;printf("[%-101s] %3d%%\r",buffer,rate);fflush(stdout);
}

使用回调函数调用

知识回顾

回调函数复习参见46.【C语言】指针(重难点)(I)文章

download函数模拟下载过程

#include "processbar.h"
typedef void (*callback_func)(int);//函数指针,返回值为void,参数是int
void download(callback_func cbf)
{int total=5000;int cur=0;while (cur<=total){usleep(8000);int rate=cur/total;cbf(rate);cur+=500;}
}int main()
{download(processbar);return 0;
}

运行结果:

5.从pv命令项目看进度条显示函数的调用

pv命令是一个开源的项目,其全称为pipe viewer,为管道查看器

官方网站:https://www.ivarch.com/programs/pv.shtml

其中有一个功能是调用进度条显示函数,以下面这个例子说说使用方法

复制文件时打印进度条显示进度

使用cp命令复制文件时不会显示复制的进度,可以使用pv命令来查看复制的进度

生成2GB的测试文件

先生成2GB的测试文件用于之后的复制

#include <stdio.h>
#include <stdlib.h>
#include <limits.h>
int main()
{FILE* ptr = fopen("test.bin", "wb+");unsigned long long size = 0;unsigned int num=0;while (size < 1024 *1024*512)//2GB{//size_t fwrite(const void *ptr, size_t size, size_t count, FILE *stream);fwrite(&num, 4, 1, ptr);num++;size++;}fclose(ptr);return 0;
}

使用pv命令复制测试文件

安装pv命令

sudo yum install pv #CentOS RedHat
sudo apt install pv #Ubuntu

从官网给出的说明来看,-p选项可以加上进度条,-e可以显示预计到达时间ETA(EstimatedTimeofArrival)

pv test.bin > test_bak.bin -p -e

 运行结果:

gif的字比较小,视频观看:

进度条演示

部分源代码分析

 以最新版本1.9.31为例分析,官网的下载速度很慢,这里给出下载链接:https://pan.baidu.com/s/19k7MR8QDKEWAm-v5RyYBXA?pwd=pxgy,提取码: pxgy 

进度条显示函数pv_display在pv-1.9.31/src/pv/display.c下

void pv_display(pvstate_t state, bool final)

 pvstate_t是自定义类型,是pvstate_s结构体的指针,而pvstate_s结构体的定义在pv-internal.h下.用于表示pv内部的状态

pvstate_s结构体的内部嵌套了几个匿名结构体,分别用于表示程序状态、输入的文件、程序控制、信号处理、Transient标记、显示状态、计算传输时需要的状态和Cursor/IPC状态等等

可以看到有之前提到的rate

http://www.lqws.cn/news/602209.html

相关文章:

  • 07CSRF 漏洞保护
  • vite项目中引入tailwindcss,难倒AI的操作
  • Modbus协议
  • 数字图像处理学习笔记
  • Spring IOC容器核心阶段解密:★Bean实例化全流程深度剖析★
  • 菜谱大全——字符串处理艺术:从文本解析到高效搜索 [特殊字符][特殊字符]
  • 城市灯光夜景人像街拍摄影后期Lr调色教程,手机滤镜PS+Lightroom预设下载!
  • 自由学习记录(66)
  • RESTful API 设计原则深度解析
  • 转录组分析流程(六):列线图
  • 笨方法学python-习题12
  • JavaScript 安装使用教程
  • 解码知识整理,使您的研究更高效!
  • 分区表设计:历史数据归档与查询加速
  • [论文阅读] 人工智能 + 软件工程 | 从软件工程视角看大语言模型:挑战与未来之路
  • python训练day46 通道注意力
  • 2025-0701学习记录19——“问题-方法-洞见”框架做汇报
  • 半导体和PN结
  • socket编程
  • Android11 添加自定义物理按键事件监听回调
  • Vite 7.0 与 Vue 3.5:前端开发的性能革命与功能升级
  • 【Linux】进程
  • NLP——RNN变体LSTM和GRU
  • Android布局管理器实战指南:从LinearLayout到ConstraintLayout的优化之旅
  • Redis——常用指令汇总指南(一)
  • 【Python】断言(assert)
  • 监听器模式
  • [Python] -基础篇8-Python中的注释与代码风格PEP8指南
  • 【C++】inline的作用
  • InnoDB数据页