【Linux系统编程】进度条的编写

目录

一,进度条的必备知识

1,缓冲区的粗略介绍

2,回车与换行

二,进度条的初步制作

1,进度条的初步矿建

2,进度条的版本一

3,进度条的版本二


一,进度条的必备知识

1,缓冲区的粗略介绍

        缓冲区是内存的一部分空间,用于临时存储输入和输出的数据。它可分为输入缓冲区和输出缓冲区。每当我们输入数据时都是往输入缓冲区中存放数据,当刷新缓冲区时,数据将会从缓冲区中拿出输入到某个变量中。每当我们输出数据时,系统将会把数据输出到输出缓冲区中,当刷新输出缓冲区时,数据将会从输出缓冲区输出到指定地方。

        其中,缓冲区的刷新时机是不同的。行缓冲会在遇到换行符时刷新,全缓冲会在缓冲区写满时刷新,而无缓冲则没有缓冲区,代表是系统调用。在C/C++中,通常用 fflush(FILE* stream) 来强制刷新指定流的缓冲区。

        C/C++中类似于sleep函数功能控制的就是缓冲区,当系统调用到sleep是,将会被缓冲区暂时保存起来,一旦sleep运行完毕之后缓冲区才刷新。进度条有时控制的就是缓冲区的刷新时间。

2,回车与换行

        “ 回车 ” 是把光标从当前位置直接指向最开头位置。“ 换行 ” 是把光标从当前位置直接指向下一行同一列的位置。我们在C语言阶段常用的 “ \n ” 指的是 换行 + 回车。而 “ \r ” 只表示回车。


二,进度条的初步制作

1,进度条的初步矿建

        首先,我们先来编写进度条的简单倒计时程序,这就需要运用回车和sleep来控制程序的运行。

#include <iostream>
#include <iomanip> //setw的头文件
#include <unistd.h> //usleep()的头文件,对应参数单位为微秒                                             
#include <cstdio>
using namespace std;
int main()
{
    int n = 10;
    while (n >= 0) {
        cout << left << setw(2) << n << '\r'; //跟C语言中printf("%-2d\r", n)效果一样
        fflush(stdout);   //强制刷新输出缓冲区
        n--;
        usleep(500000);    //这里我们控制缓冲时间为0.5秒
    }
    cout << endl;
    return 0;
}

        下一步,要思考进度条的框架设计。这里的进度条将外围用 " = " 表示进度的加载,外围设置了百分比显示加载数据。用 "|/-\" 来表示其中的加载,即顺时针旋转。

2,进度条的版本一

        首先,外面设置一个头文件 "process.h" 进行必要的设置

#include <iostream>
#include <string>
#include <unistd.h>
#include <iomanip>
#include <cstdio>
using namespace std;

#define Body '='   //使用body来表示进度
#define Head '>'   //Head表示目前加载的终点,这里用 ' > ' 表示

void process1();  //进度条函数

        下面,进行进度功能的编写。这里使用 usleep 功能来控制进度的的运行,这里需注意的是输出缓冲区的刷新。

void process1()
{

    //用lable表示进度条的加载
    string lable("|/-\\");  
    string nums;
    int count = 0;
    int lablesize = lable.size();
    nums.push_back(Head);
    while (count <= 100)
    {
        cout << "[" << left << setw(100) << nums << "]";
        cout << "[" << "%" << count << "]";
        cout << "[" << lable[count % lablesize] << "]" << '\r';
        fflush(stdout);
        nums.clear();
        count++;
        nums.append(count, Body);
        if (count < 100)
        {
            nums.push_back(Head);
        }

        //这里我们设置每0.6秒加载一次
        usleep(60000); 
    }
    cout << endl;
}

运行最终结果:

[====================================================================================================][%100][|]

3,进度条的版本二

        进度条一般都是运用在一种应用上,表示应用的加载过程。很显然,版本一的进度条只是无脑运行,不知道程序进度是多少,即没有依附应用进度,比如下载程序,这时的进度条需依附于下载进度来跟进。

        头文件 "process.h" 添加如下:

#include <iostream>
#include <string>
#include <unistd.h>
#include <iomanip>
#include <cstdio>
#include <cstdlib>
using namespace std;

#define Body '='
#define Head '>'
#define Max 103
#define FileSize 1024*1024*1024  //设置FileSize文件内存为1G,表示下载程序的总大小

typedef void (*callback_t)(double);  //利用函数指针来进行封装进度运用


void download(callback_t);  //模拟一种下载进度
void process2(double rate); //进度条跟进程序

        这里,在设置download下载时要将每一次的下载进度传递给进度条让其显示百分比。

void download(callback_t cb)   //利用回调函数的形式设置进度
{
    srand(time(0)*1024);
    int total = FileSize;
    while (total)
    {
        //下面表示一次下载动作
        usleep(10000);
        int one = rand() % (1024 * 1024 * 5);
        total -= one;
        if (total < 0) 
        {
            total = 0;
        }
        //表示当前的进度
        int download = FileSize - total;
        double rate = (download * 1.0 / (FileSize)) * 100.0;
        cb(rate); //每一次进度条的传递
    }
}

进度条设置时要说明以下几点:

        1,我们使用 "|/-\\" 表示进度跟进时是根据下载进度进行的,与当前的进度无关。

        2,进度条的总设置需与下载程序紧紧联系。比如当程序加载完时,“ > ” 进度条中表示进          度运行的就要停止,即删除。

        3,在输出进度运行过程,我们可添加其色彩表示美观,链接:色彩文本的增添

void process2(double rate)
{

    //用lable表示下载任务一直在跟进
    string lable("|/-\\"); 

    //注意,这里要保留之前的进度,需设置静态
    static char buffer[Max] = { 0 };
    static int cnt = 0;
    if (rate <= 1.0)
    {
        buffer[0] = Head;
    }
    printf("\033[1;31;46m[%-100s]\033[0m[%.1lf%%][%c]\r", buffer, rate, lable[cnt % lable.size()]);  //设置色彩,这里我们设置高亮/加粗,青色背景,红色字体的色彩

    fflush(stdout);

    //下面控制进度的跟进
    buffer[(int)rate] = Body;
    if ((int)rate + 1 < 100)
    {
        buffer[(int)(rate + 1)] = Head;
    }
    if (rate >= 100.0)
    {
        cout << endl;
    }
    cnt++;
    cnt %= lable.size();
}

总代码如下:

#include "process.h"
//版本一
void process1()
{string lable("|/-\\");string nums;int count = 0;int lablesize = lable.size();nums.push_back(Head);while (count <= 100){cout << "[" << left << setw(100) << nums << "]";cout << "[" << "%" << count << "]";cout << "[" << lable[count % lablesize] << "]" << '\r';fflush(stdout);nums.clear();count++;nums.append(count, Body);if (count < 100){nums.push_back(Head);}usleep(60000);}cout << endl;
}
//版本二
void download(callback_t cb)
{srand(time(0) * 1024);int total = FileSize;while (total){usleep(10000);int one = rand() % (1024 * 1024 * 5);total -= one;if (total < 0){total = 0;}int download = FileSize - total;double rate = (download * 1.0 / (FileSize)) * 100.0;cb(rate);}
}
void process2(double rate)
{static string lable("|/-\\");static char buffer[Max] = { 0 };static int cnt = 0;if (rate <= 1.0){buffer[0] = Head;}printf("\033[1;31;46m[%-100s]\033[0m[%.1lf%%][%c]\r", buffer, rate, lable[cnt % lable.size()]);fflush(stdout);buffer[(int)rate] = Body;if ((int)rate + 1 < 100){buffer[(int)(rate + 1)] = Head;}if (rate >= 100.0){cout << endl;}cnt++;cnt %= lable.size();
}
int main()
{//process1(); //使用进度条粗略版本一download(process2); //使用进度条进化版本二return 0;
}

        最后,要说明的是,以上程序都是在Linux系统下运行进行的,在VS或其它编译器下可能会出现错误消息,这时因为不同平台支持的C标准或系统设置不同而造成的差异。

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

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

相关文章

详细了解STM32----GPIO

提示&#xff1a;永远支持免费开源知识文档&#xff0c;喜欢的点个关注吧&#xff01;谢谢&#xff01; 文章目录 一、什么是GPIO&#xff1f;二、GPIO基本结构三、GPIO的输入输出模式1、推挽输出2、开漏输出3、复用推挽4、复用开漏1、浮空输入2、上拉输入&#xff13;、下拉输…

FastAPI之嵌套模型

请求体 - 嵌套模型 使用 FastAPI&#xff0c;你可以很随意的实现模型的嵌套、定义、校验、记录文档&#xff0c;并使用任意深度嵌套的模型&#xff0c;这其实都是FastAPI的核心模块P一单提成进行做的。。 List 字段 from fastapi import FastAPI from pydantic import BaseM…

基于JavaWeb+SSM+Vue童装商城小程序系统的设计和实现

基于JavaWebSSMVue童装商城小程序系统的设计和实现 源码获取入口Lun文目录前言主要技术系统设计功能截图订阅经典源码专栏Java项目精品实战案例《500套》 源码获取 源码获取入口 Lun文目录 目 录 摘 要 III Abstract 1 1 系统概述 2 1.1 概述 3 1.2课题意义 4 1.3 主要内容 5…

BearPi Std 板从入门到放弃 - 先天篇(1)(阶段 : 智慧城市 - 智慧路灯)

简介 对前面几篇整合, 做个小小汇总试验, 使用BearPi E53_SC1扩展板主芯片: STM32L431RCT6串口: Usart1扩展板与主板连接: I2C : I2C1 (光照强度传感器&#xff1a;BH1750)LED: PB9步骤 创建项目 参考 BearPi Std 板从入门到放弃 - 引气入体篇&#xff08;1&#xff09;(由零创…

【测试人生】数据同步和迁移的变更注意事项

数据同步或者迁移操作也算是线上数据变更的一种类型。由于涉及的数据量非常大&#xff0c;一旦发生故障&#xff0c;会直接影响线上业务&#xff0c;并且较难止损。从变更风险管控的角度考虑&#xff0c;数据同步或迁移操作也需要走合理的发布窗口&#xff0c;并且在操作前也需…

浅谈Google Play ASO 优化

什么是ASO ASO即APP Store Optimization&#xff0c;是用于提高APP在应用市场排名的工具&#xff0c;其实也就是移动产品的SEO工作。 ASO是为了提高该产品的搜索结果成绩&#xff0c;提升APP的下载量&#xff0c;针对Google Play来说&#xff0c;ASO就是优化APP页面。 为什么…

Linux升级nginx版本

处于漏洞修复目的服务器所用nginx是1.16.0版本扫出来存在安全隐患&#xff0c;需要我们升级到1.17.7以上。 一般nginx默认在 /usr/local/ 目录&#xff0c;这里我的nginx是自定义的路径安装在 /app/weblogic/nginx 。 1.查看生产环境nginx版本 cd /app/weblogic/nginx/sbin/…

Redis基础入门

第1章&#xff1a;引言 大家好&#xff01;我是小黑&#xff0c;今天咱们来聊聊Redis。Redis&#xff0c;这个名字你可能在不少地方听过&#xff0c;尤其是在后端开发领域&#xff0c;它可是个大名鼎鼎的角色。&#xff0c;Redis是一个开源的内存中数据结构存储系统&#xff0…

放弃原生SQL:Python中更优雅的数据库操作

概要 在Python中&#xff0c;通过原生SQL语句进行数据库操作是一种传统的方式&#xff0c;但现代的Python开发中&#xff0c;使用ORM&#xff08;Object-Relational Mapping&#xff09;工具和数据库连接库可以更加高效和优雅地进行增删改查操作。本文将详细介绍Python中放弃原…

解决IDEA中多个项目不在同一窗口下显示的问题和添加新的git的URL

以上是添加显示多个项目 以下是给新添加的项目添加git

LeetCode算法题解(单调栈)|LeetCode84. 柱状图中最大的矩形

一、LeetCode84. 柱状图中最大的矩形 题目链接&#xff1a;84. 柱状图中最大的矩形 题目描述&#xff1a; 给定 n 个非负整数&#xff0c;用来表示柱状图中各个柱子的高度。每个柱子彼此相邻&#xff0c;且宽度为 1 。 求在该柱状图中&#xff0c;能够勾勒出来的矩形的最大…

做外贸很多时候还是要学会随机应变

马上又要到年底了&#xff0c;相信已经有一部分小伙伴开启了催单模式&#xff0c;希望客户尽量在春节前将订单落实下来&#xff0c;自然也有很多客户会在春节前的这一段时间开始陆续拜访自己观望了很久的工厂。 其实对于贸易公司来说&#xff0c;对于来看工厂的客户&#xff0…

ChatGPT,作为一种强大的自然语言处理模型,具备显著优势,能够帮助您在各个领域取得突破

2023年随着OpenAI开发者大会的召开&#xff0c;最重磅更新当属GPTs&#xff0c;多模态API&#xff0c;未来自定义专属的GPT。微软创始人比尔盖茨称ChatGPT的出现有着重大历史意义&#xff0c;不亚于互联网和个人电脑的问世。360创始人周鸿祎认为未来各行各业如果不能搭上这班车…

Kotlin 作用域函数:理解 apply, let, 和 with

Kotlin提供了几个作用域函数来优化和简化代码的结构。 本文将对比分析 apply, let, 和 with 三个函数。 一、对比分析&#xff1a; apply&#xff1a;在其接收者的上下文中执行代码块&#xff0c;并返回接收者对象。let&#xff1a;在其接收者的上下文中执行代码块&#xff…

Kotlin:内置函数let、also、with、run、apply

前言 在Kotlin中&#xff0c;有一些用于扩展 & 方便开发者编码的内置函数&#xff0c;能大大提高开发者的开发效率。今天&#xff0c;我将主要讲解的是&#xff1a; let函数also函数with函数run函数apply函数 基础知识&#xff1a;接口回调中Lambda使用 在Kotlin中可使用…

栈和队列的互相实现

用队列实现栈 OJ链接 请你仅使用两个队列实现一个后入先出&#xff08;LIFO&#xff09;的栈&#xff0c;并支持普通栈的全部四种操作&#xff08;push、top、pop 和 empty&#xff09;。 实现 MyStack 类&#xff1a; void push(int x) 将元素 x 压入栈顶。int pop() 移除并返…

Mybatis XML增删操作(结合上文)

先来"增"操作 在UserInfoXMLMapper.xml里面写 <?xml version"1.0" encoding"UTF-8"?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <…

nginx多端口部署

1.配置nginx.conf文件 有几个端口需要部署就写几个server&#xff0c;我这里只部署了两个端口分别为80和81端口&#xff0c;所以有两个server文件。80端口项目入口在根目录的test文件中&#xff0c;81端口项目入口在根目录的test1文件夹中。 2.准备项目文件html文件 在/test1…

Dockerfile部署Java项目挂载使用外部配置文件

Dockerfile部署Java项目挂载使用外部配置文件 技术博客 http://idea.coderyj.com/ 需求是由于java项目使用的是nacos 而且每次部署nacos服务器ip不一样导致要重新打包,想引入外部配置文件进行打包 1.需求是由于java项目使用的是nacos 而且每次部署nacos服务器ip不一样导致要重新…

数据结构和算法专题---5、调度算法与应用

本章我们会对调度算法做个简单介绍&#xff0c;包括常用的调度算法&#xff08;FCFS、SJF、RR、HPF&#xff09;的概述、实现方式、典型场景做个说明。 什么是调度算法 调度算法常见于操作系统中&#xff0c;因为系统资源有限&#xff0c;当有多个进程&#xff08;或多个进程…