Linux第一个系统程序---进度条

进度条---命令行版本

回车换行

其实本质上回车和换行是不同概念,我们用一张图来简单的理解一下:

在计算机语言当中:

换行符:\n

回车符:\r

\r\n:回车换行

这时候有人可能会有疑问:我在学习C/C++语言的时候,单纯的\n就起到了回车加换行的行为呀?!

答:因为在语言层面\n给我们编译成了\r\n 

缓冲区问题

在我们目前的阶段,我们将缓冲区看作是一段内存块,我们可以先实现一个测试代码:

#touch test.c#可以直接
vim test.c

测试代码

技巧补充:我们可以在底行模式使用man手册查看sleep(3)函数所需的头文件:
:!man 3 sleep

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

 我们在Linux编译后./test执行时,当只有上被注释的代码被执行,和只有第二句printf的代码,我们发现前者内容打印出来,程序3秒后结束,后者是3秒后程序结束才打印内容,我们可以明显感受到程序是先执行sleep(3)的,但是我们之前学习C/C++知道,我们定义的程序逻辑都是从上往下执行的,可是我们为什么没有看到字符串呢?

我们应该知道,对于后者,printf在sleep之前一定执行完了,但是显示器上没有显示,那么在我们休眠3秒期间,字符串" hello Linux! "在哪里?其实该字符串在缓冲区里面,可以理解为内存有一小内存块,将该字符串临时保存在该处了。我们带有\n的字符串可以在显示屏上打印,我们称为行刷新,也就是遇到\n,printf执行完就默认将其字符串直接显示到显示器了(行刷新),可以理解为没有\n就不给行刷新;

那后者为什么最后也可以刷新出来呢?

是因为程序退出了,会自动刷新缓冲区!


我们如何可以让不带\n的字符串能够立马刷新呢?

我们如果想让不带\n的字符串立马刷新,可以使用fflush:

fflush 是 C 语言标准库函数之一,用于刷新流的缓冲区。其原型定义在 <stdio.h> 头文件中

我们可以进行man手册查看。

printf("hello Linux!");
fflush(stdout);

其实我们的程序会默认打开三个输入输出流,分别是:(描述符分别为1,2,3)

  1. 标准输入流(stdin:用于从标准输入设备(通常是键盘)读取数据。

  2. 标准输出流(stdout:用于向标准输出设备(通常是屏幕)输出数据。

  3. 标准错误流(stderr:用于输出错误信息和其他诊断信息,通常也是输出到屏幕。

都是一个指向 FILE 结构体的指针,定义在 <stdio.h> 中。

我们的printf默认是从标准输出流里面打,我们可以看到fprintf其实就是比printf多了一个参数:

在Linux下,一切皆文件,当一个程序在 Linux 下运行时,操作系统会自动为该程序打开三个标准流文件描述符,也就是下面两条代码是等价的,fprintf只是显示的表示出往哪里输出:

    fprintf(stdout,"hello Linux!\n");printf("hello Linux!\n");

 我们就可以利用缓冲区来实现一个倒计时:光标显示后回退(回车)

int main()
{int i=9;while(i>=0){printf("%d\r",i);i--;sleep(1);}return 0;
}

但是我们发现在显示器上没有显示任何数据,这是为什么呢?

因为我们行刷新是需要\n的,%d后是\r,\r不支持我们行刷新,所以对应的信息并未有显示出来,还在缓冲区里面存着呢,所以需要用到fflush(stdout);

int main()
{int i=9;while(i>=0){printf("%d\r",i);fflush(stdout);i--;sleep(1);}printf("\n");//跑完之后想要保留0(命令行),不让会覆盖return 0;
}

为了使我们对下面进度条程序更好的实现,我们来看看当i取10的时候:

命令行变化:10-90-80-70-60-50-40-30-20-10-00----00

这时候,我们应该好好理解显示:
举个例子,我们往显示器上输入:12345,是输出的是12345数字,还是'1' '2' '3' '4' '5'字符?

显示器只认识字符!显示器是字符设备,所以输出的是后者,这也是为什么我们需要像%d的格式化输出,所以我们可以将上述代码中的%d改成%2d,还可以在前面加一个-即"%-2d"来整体靠左显示,代码我就不写出来了,在原代码上做改进就🆗。

以上我们进度条的预备工作就基本完成了(回车换行,缓冲区问题,格式问题,字符设备的理解问题,输出设备的刷新问题)

我们想写一个怎么样的进度条

框架的搭建

创建文件

首先,在我们自己的工作目录中创建四个文件:

  • main.c

  • Makefile

  • process.c

  • process.h

编写 Makefile

使用 vim 编辑器打开 Makefile,并写入以下内容:

SRC:=$(wildcard *.c)
OBJ:=$(SRC:.c=.o)
BIN=processbar$(BIN): $(OBJ)gcc -o $@ $^%.o: %.cgcc -c $<.PHONY: clean
clean:rm -f $(OBJ) $(BIN)

这个 Makefile 定义了如何编译和链接你的程序。它使用 wildcard 自动查找所有 .c 文件,并生成相应的 .o 文件。最终生成的可执行文件名为 processbar

编写 process.h

使用 vim 编辑器打开 process.h,并写入以下内容:

#pragma once
#include <stdio.h>void process_v1();

这个头文件声明了一个函数 process_v1,该函数将在 process.c 中实现。

编写 process.c

使用 vim 编辑器打开 process.c,并写入以下内容:

#include "process.h"void process_v1() {printf("hello rose!\n");
}

这个文件实现了 process_v1 函数,目前只是简单地打印一条消息。

编写 main.c

使用 vim 编辑器打开 main.c,并写入以下内容:

#include "process.h"int main() {process_v1();return 0;
}

这个文件是程序的入口点,它调用了 process_v1 函数。

编译和运行

完成上述文件编写后,你可以使用以下命令进行编译:

make

这将根据 Makefile 的规则编译并生成 processbar 可执行文件。然后,你可以运行该程序:

./processbar

如果一切顺利,你应该会看到输出 "hello rose!"。

清理

最后,使用以下命令清理生成的文件:

make clean

这将删除所有 .o 文件和 processbar 可执行文件。

接下来,我们就可以在 process.c 中实现进度条的功能。可以根据具体需求设计进度条的更新逻辑和显示方式。

version1原理版本

#include"process.h"
#include<string.h>
#include<unistd.h>#define NUM 101//因为会有一个\0的存在
#define STYLE '='void process_v1()
{    char buffer[NUM];memset(buffer,0,sizeof(buffer));const char *lable="|/-\\";size_t len=strlen(lable);//计数器int cnt=0;while(cnt<=100)//这个循环会循环101次{printf("[%-100s][%d%%][%c]\r",buffer,cnt,lable[cnt%len]);fflush(stdout);buffer[cnt]=STYLE;cnt++;//sleep(1);usleep(10000);}printf("\n");return;
}
  1. 定义常量和变量

    • NUM 定义为 101,因为进度条的长度为 100 个字符,加上一个字符串的终止符 \0
    • STYLE 定义为 =,用于表示进度条的已填充部分。
    • buffer 是一个字符数组,用于存储进度条的当前状态。
    • lable 是一个字符串,包含旋转的符号 |/-\,用于在进度条旁边显示一个旋转的动画效果。
    • len 是 lable 字符串的长度。
    • cnt 是一个计数器,用于控制进度条的更新和旋转符号的切换。
  2. 初始化缓冲区

    • 使用 memset 函数将 buffer 初始化为全 0,即空字符串。
  3. 进度条更新循环

    • while 循环会执行 101 次,因为 cnt 从 0 开始,直到 100 结束。
    • 在每次循环中,使用 printf 函数输出当前的进度条状态:
      • [%-100s] 表示一个宽度为 100 的左对齐字符串,buffer 作为参数传入,表示进度条的当前填充状态。
      • [%d%%] 表示当前的百分比进度,cnt 作为参数传入。
      • [%c] 表示旋转符号,lable[cnt%len] 计算当前应该显示的旋转符号。
    • 使用 \r 作为 printf 的结尾,表示返回到行首,这样下一次输出会覆盖当前行的内容。
    • 调用 fflush(stdout) 确保输出立即显示在屏幕上,而不是被缓冲。
    • 将 buffer[cnt] 设置为 STYLE,即 =,表示进度条的已填充部分向右扩展一个字符。
    • cnt++ 更新计数器。
    • usleep(10000) 暂停 10 毫秒,使进度条的更新速度适中,便于观察。注释掉的 sleep(1) 是暂停 1 秒的另一种方式,但会使进度条更新过慢。
  4. 结束输出

    • 循环结束后,调用 printf("\n") 输出一个换行符,使光标移动到下一行,避免后续输出覆盖进度条。

整体功能:

这个 process_v1 函数实现了一个动态更新的文本进度条,进度条的长度为 100 个字符,旁边有一个旋转的符号动画。通过在循环中逐步填充 buffer 并输出,模拟了进度条的动态增长过程。每次更新后,使用 \r 返回行首并刷新输出,实现了进度条的原地更新效果。

version2真实版本

main.c

#include "process.h"#include <stdio.h>
#include <unistd.h>double total = 1024.0; // 总下载量,单位为MB
double speed = 1.0;    // 下载速度,单位为MB/svoid DownLoad() {double current = 0; // 当前已下载量while (current <= total) {// 刷新进度条FlushProcess(total, current);// 模拟下载数据,每次循环下载speed大小的数据usleep(300000); // 暂停300ms,模拟下载延时current += speed;}printf("\ndownload %.2lfMB Done\n", current);
}int main() {DownLoad();return 0;
}

version2

#include "process.h"
#include <string.h>
#include <unistd.h>#define NUM 101 // 因为会有一个\0的存在
#define STYLE '='// version2
void FlushProcess(double total, double current) {char buffer[NUM];memset(buffer, 0, sizeof(buffer));const char *lable = "|/-\\";size_t len = strlen(lable);static int cnt = 0; // 静态变量,用于记录旋转符号的位置// 计算当前进度条的填充数量int num = (int)(current * 100 / total);for (int i = 0; i < num; i++) {buffer[i] = STYLE;}double rate = current / total; // 计算当前下载进度的百分比cnt %= len;                    // 计算旋转符号的索引printf("[%-100s][%.1f%%][%c]\r", buffer, rate * 100, lable[cnt]);cnt++;fflush(stdout); // 确保进度条立即显示
}

进度条与下载模拟功能梳理:

  • 下载模拟

    • DownLoad 函数模拟了一个下载过程,其中 total 表示总下载量,speed 表示下载速度。
    • 在 while 循环中,每次循环模拟下载 speed 大小的数据,并通过 usleep 函数暂停一段时间来模拟下载延时。
    • 循环过程中,不断调用 FlushProcess 函数来刷新进度条,显示当前的下载进度。
  • 进度条刷新

    • FlushProcess 函数负责根据当前下载量 current 和总下载量 total 计算进度条的填充状态,并输出到屏幕。
    • buffer 数组用于存储进度条的当前状态,num 变量计算出需要填充的 = 字符的数量。
    • 使用 printf 函数输出进度条,其中 [%-100s] 表示一个宽度为 100 的左对齐字符串,用于显示进度条的填充部分;[%.1f%%] 显示当前的下载百分比;[%c] 显示一个旋转的符号,用于增加动画效果。
    • 使用 \r 作为 printf 的结尾,表示返回到行首,使下一次输出覆盖当前行的内容,实现进度条的原地更新效果。
    • fflush(stdout) 确保进度条的输出立即显示在屏幕上,而不是被缓冲。

整体功能

这个程序通过模拟下载过程,并在每次下载数据后刷新进度条,实现了动态显示下载进度的效果。进度条的长度为 100 个字符,旁边有一个旋转的符号动画,使用户可以直观地看到下载进度的变化。当下载完成后,程序会输出一条完成信息,告知用户下载已成功完成。

我们不仅仅可以在下载方面使用进度条,我们也可以在上传方面使用进度条,因此,在main.c源文件当中,我们typedef一个函数指针来使用回调函数来优化代码:

#include"process.h"#include<stdio.h>
#include<unistd.h>typedef void (*callback_t) (double total,double current);//函数指针double total=1024.0;
double speed=1.0;//回调函数
void DownLoad(callback_t cb)
{double current=0;while(current<=total){//刷新进度cb(total,current);//下载代码usleep(3000);//充当下载数据current+=speed;}printf("\ndownload %.2lfMB Done\n",current);
}void UpLoad(callback_t cb)
{double current=0;while(current<=total){//刷新进度cb(total,current);//下载代码usleep(3000);//充当下载数据current+=speed;}printf("\nUpload %.2lfMB Done\n",current);
}
int main()
{DownLoad(FlushProcess);UpLoad(FlushProcess);return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

西电-神经网络基础与应用-复习笔记

此为24年秋研究生课程复习笔记 导论 神经网络的研究方法分为 连接主义&#xff0c;生理学派&#xff0c;模拟神经计算。高度的并行、分布性&#xff0c;很强的鲁棒和容错性。便于实现人脑的感知功能(音频图像的识别和处理)。符号主义&#xff0c;心理学派&#xff0c;基于符号…

利用obs studio制作(人像+屏幕)录制影像

1.什么是obs? OBS&#xff08;Open Broadcaster Software&#xff09;是一款功能强大的开源软件&#xff0c;它使用户能够直接从电脑录制视频和直播内容到 Twitch&#xff0c;YouTube 和 Facebook Live 等平台。它在需要直播或录制屏幕活动的游戏玩家、YouTube 用户和专业人士…

maven多模块项目编译一直报Failure to find com.xxx.xxx:xxx-xxx-xxx:pom:1.0-SNAPSHOT in问题

工作中项目上因为多版本迭代&#xff0c;需要对不同迭代版本升级版本号&#xff0c;且因为项目工程本身是多模块结构&#xff0c;且依然多个其他模块工程。 在将工程中子模块的pom.xml中版本号使用变量引用父模块中定义的版本号时&#xff0c;一直报Failure to find com.xxx.x…

音视频入门基础:RTP专题(2)——使用FFmpeg命令生成RTP流

通过FFmpeg命令可以将一个媒体文件转推RTP&#xff1a; ffmpeg -re -stream_loop -1 -i input.mp4 -c:v copy -an -f rtp rtp://192.168.0.102:5400 但是通过ffplay尝试播放上述产生的RTP流时会报错&#xff1a;“Unable to receive RTP payload type 96 without an SDP file …

Nacos 3.0 Alpha 发布,在安全、泛用、云原生更进一步

自 2021 年发布以来&#xff0c;Nacos 2.0 在社区的支持下已走过近三年&#xff0c;期间取得了诸多成就。在高性能与易扩展性方面&#xff0c;Nacos 2.0 取得了显著进展&#xff0c;同时在易用性和安全性上也不断提升。想了解更多详细信息&#xff0c;欢迎阅读我们之前发布的回…

C语言gdb调试

目录 1.gdb介绍 2.设置断点 2.1.测试代码 2.2.设置函数断点 2.3.设置文件行号断点 2.4.设置条件断点 2.5.多线程调试 3.删除断点 3.1.删除指定断点 3.2.删除全部断点 4.查看变量信息 4.1.p命令 4.2.display命令 4.3.watch命令 5.coredump日志 6.总结 1.gdb介绍…

【xLua】xLua-master签名、加密Lua文件

GitHub - Tencent/xLua: xLua is a lua programming solution for C# ( Unity, .Net, Mono) , it supports android, ios, windows, linux, osx, etc. 如果你想在项目工程上操作&#xff0c;又发现项目工程并没导入Tools&#xff0c;可以从xLua-master工程拷贝到项目工程Assets…

9.4 visualStudio 2022 配置 cuda 和 torch (c++)

一、配置torch 1.Libtorch下载 该内容看了【Libtorch 一】libtorchwin10环境配置_vsixtorch-CSDN博客的博客&#xff0c;作为笔记用。我自己搭建后可以正常运行。 下载地址为windows系统下各种LibTorch下载地址_libtorch 百度云-CSDN博客 下载解压后的目录为&#xff1a; 2.vs…

Python基于YOLOv8和OpenCV实现车道线和车辆检测

使用YOLOv8&#xff08;You Only Look Once&#xff09;和OpenCV实现车道线和车辆检测&#xff0c;目标是创建一个可以检测道路上的车道并识别车辆的系统&#xff0c;并估计它们与摄像头的距离。该项目结合了计算机视觉技术和深度学习物体检测。 1、系统主要功能 车道检测&am…

相加交互效应函数发布—适用于逻辑回归、cox回归、glmm模型、gee模型

在统计分析中交互作用是指某因素的作用随其他因素水平变化而变化&#xff0c;两因素共同作用不等于两因素单独作用之和(相加交互作用)或之积(相乘交互作用)。相互作用的评估是尺度相关的&#xff1a;乘法或加法。乘法尺度上的相互作用意味着两次暴露的综合效应大于&#xff08;…

ECharts饼图下钻

背景 项目上需要对Echarts饼图进行功能定制&#xff0c;实现点击颜色块&#xff0c;下钻显示下一层级占比 说明 饼图实现点击下钻/面包屑返回的功能 实现 数据结构 [{name: a,value: 1,children: [...]},... ]点击下钻 // 为图表绑定点击事件&#xff08;需要在destroy…

MySQL-事务

事务特性 在关系型数据库管理系统中&#xff0c;事务必须满足 4 个特性&#xff0c;即所谓的 ACID。 原子性&#xff08;Atomicity&#xff09; 事务是一个原子操作单元&#xff0c;其对数据的修改&#xff0c;要么全都执行&#xff0c;要么全都不执行。 修改操作>修改B…

C# 元组

总目录 C# 语法总目录 C# 元组 C# 介绍元组1. 元组元素命名2. 元组的解构3. 元组的比较 总结参考链接 C# 介绍 C#主要应用于桌面应用程序开发、Web应用程序开发、移动应用程序开发、游戏开发、云和服务开发、数据库开发、科学计算、物联网&#xff08;IoT&#xff09;应用程序、…

用 Python 绘制可爱的招财猫

✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连✨ ​​​​​ ​​​​​​​​​ ​​​​ 招财猫&#xff0c;也被称为“幸运猫”&#xff0c;是一种象征财富和好运的吉祥物&#xff0c;经常…

Java多线程

一、线程的简介: 1.普通方法调用和多线程: 2.程序、进程和线程: 在操作系统中运行的程序就是进程&#xff0c;一个进程可以有多个线程 程序是指令和数据的有序集合&#xff0c;其本身没有任何运行的含义&#xff0c;是一个静态的概念&#xff1b; 进程则是执行程序的一次执…

IP 地址与蜜罐技术

基于IP的地址的蜜罐技术是一种主动防御策略&#xff0c;它能够通过在网络上布置的一些看似正常没问题的IP地址来吸引恶意者的注意&#xff0c;将恶意者引导到预先布置好的伪装的目标之中。 如何实现蜜罐技术 当恶意攻击者在网络中四处扫描&#xff0c;寻找可入侵的目标时&…

鸿蒙面试 2025-01-09

鸿蒙分布式理念&#xff1f;&#xff08;个人认为理解就好&#xff09; 鸿蒙操作系统的分布式理念主要体现在其独特的“流转”能力和相关的分布式操作上。在鸿蒙系统中&#xff0c;“流转”是指涉多端的分布式操作&#xff0c;它打破了设备之间的界限&#xff0c;实现了多设备…

GDPU Android移动应用 重点习题集

目录 程序填空 ppt摘选 题目摘选 “就这两页ppt&#xff0c;你还背不了吗” “。。。” 打开ppt后 “Sorry咯&#xff0c;还真背不了&#x1f61c;” 更新日志 考后的更新日志 没想到重点勾了一堆&#xff0c;还愣是没考到其中的内容&#xff0c;翻了一下&#xff0c;原…

Unity3d 基于Barracuda推理库和YOLO算法实现对象检测功能

前言 近年来&#xff0c;随着AI技术的发展&#xff0c;在游戏引擎中实现和运行机器学习模型的需求也逐渐显现。Unity3d引擎官方推出深度学习推理框架–Barracuda &#xff0c;旨在帮助开发者在Unity3d中轻松地实现和运行机器学习模型&#xff0c;它的主要功能是支持在 Unity 中…

【Notepad++】Notepad++如何删除包含某个字符串所在的行

Notepad如何删除包含某个字符串所在的行 一&#xff0c;简介二&#xff0c;操作方法三&#xff0c;总结 一&#xff0c;简介 在使用beyoundcompare软件进行对比的时候&#xff0c;常常会出现一些无关紧要的地方&#xff0c;且所在行的内容是变化的&#xff0c;不方便进行比较&…