Linux第一个小程序——进度条

预备知识------缓冲区与换行

先把重要结论放这:

  1. 数据的输入输出不会马上到达目的地,会先呆在缓冲区。
  2. 缓冲区相当于一行数据空间。
  3. 回车时,光标回到开头,相当于清空原有数据。
  4. 换行时,会直接将缓冲区数据输出。
  5. 程序结束时,会强制将缓冲区的数据输出。

详情见我的另一篇文章:

缓冲区与回车换行-CSDN博客

写一个倒计时程序

cpp 复制代码
#include<stdio.h>
#include<unistd.h>

int main()
{
  int i = 10;
  while(i)
  {
    printf("%-2d\r", i);
    fflush(stdout);
    sleep(1);
    i--;
  }
  return 0;
}

%-2d,保留两位数且向左对齐

编写入门版进度条

cpp 复制代码
#include<stdio.h>
#include<unistd.h>
#include<string.h>

#define MAX 101
#define LABEL '='

int main()
{
  char bar[MAX];
  memset(bar, '\0', sizeof(bar));
  int count = 101;
  while(count--)
  {
    bar[100 - count] = LABEL;
    usleep(100000);
    printf("%s\r", bar);
    fflush(stdout);
  }
  printf("\n");
  return 0;
}

多文件编写升级版进度条

Version1

main.c

cpp 复制代码
#include"processbar.h"

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

processbar.h

cpp 复制代码
#pragma once

#include<stdio.h>

#define NUM 101
#define Body '='
#define Head '>'

void processbar();

processbar.c

cpp 复制代码
#include"processbar.h"
#include<unistd.h>
#include<string.h>

const char *loading = "|/-\\";

void processbar()
{
  char bar[NUM];
  int n = strlen(loading);
  memset(bar, '\0', sizeof(bar));
  int cnt = 0;
  while(cnt <= 100)
  {
    bar[cnt++] = Body;
    if(cnt < 100)
      bar[cnt] = Head;
    printf("[%-101s][%3d%%][%c]\r", bar, cnt - 1, loading[cnt%n]);
    fflush(stdout);
    usleep(100000);
  }
  printf("\n");
}

makefile

cpp 复制代码
processbar:main.o  processbar.o
	gcc -o $@ $^

main.o:main.c
	gcc -c main.c

processbar.o:processbar.c
	gcc -c processbar.c

.PHONY:clean

clean:
	rm processbar main.o processbar.o

Version2

main.c

cpp 复制代码
#include"processbar.h"

int main()
{
  processbar();

  return 0;
}

processbar.h

cpp 复制代码
#pragma once

#include<stdio.h>
#include<unistd.h>
#include<string.h>
#include<stdlib.h>
#include<time.h>

#define NUM 101
#define Body '='
#define Head '>'
#define FILESIZE 1024*1024*1024

void processbar();

processbar.c

cpp 复制代码
#include"processbar.h"

char bar[NUM];
const char *label = "|/-\\";

void processbar()
{
  int total = FILESIZE;
  int num = 0;
  srand(time(NULL)^1023);
  memset(bar, '\0', sizeof(bar));
  while(total)
  {
    usleep(100000);
    int one = rand()%(1024*1024*50);
    total -= one;
    if(total < 0) total = 0;
    double rate = ((FILESIZE - total)*1.0 / (FILESIZE)) * 100.0;
    num++; num %= 4;
    memset(bar, Body, sizeof(char)*((int)rate + 1));
    if((int)rate < 100) bar[(int)rate] = Head;
    printf("[%-101s][%6.2f%%][%c]\r", bar, rate, label[num]);
    fflush(stdout);
  }
  printf("\n");
}

makefile

cpp 复制代码
processbar:main.o  processbar.o
	gcc -o $@ $^

main.o:main.c
	gcc -c main.c

processbar.o:processbar.c
	gcc -c processbar.c

.PHONY:clean

clean:
	rm processbar main.o processbar.o
相关推荐
在角落发呆38 分钟前
Linux转发配置:解锁网络互联的核心密码
linux·运维·网络
齐潇宇1 小时前
Zabbix 7 概述与配置
linux·zabbix·监控告警
裴东青2 小时前
10-实战:RuoYi-Cloud的自动化发布
运维·ci/cd·自动化
江公望2 小时前
Ubuntu htop命令,10分钟讲清楚
linux·服务器
哎呦,帅小伙哦2 小时前
Linux 时间:从原子钟到 clock_gettime 的每一面
linux·运维·服务器
sxgzzn3 小时前
新能源场站数智化转型:基于数字孪生与AI的智慧运维管理平台解析
大数据·运维·人工智能
张小姐的猫3 小时前
【Linux】多线程 —— 线程互斥
linux·运维·服务器·c++
CodeMartain3 小时前
Dify Windows 原生部署(无 Docker、纯本地)
运维·docker·容器
xxx1x1x3 小时前
极客向:DLL/运行库故障的底层逻辑与自动化修复方案
运维·自动化·dll文件·dll·dll修复·dll缺失·dll一键修复
YuanDaima20483 小时前
Linux 进阶运维与 AI 环境实战:进程管理、网络排错与 GPU 监控
linux·运维·服务器·网络·人工智能