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
相关推荐
小白跃升坊22 分钟前
基于1Panel的AI运维
linux·运维·人工智能·ai大模型·教学·ai agent
跃渊Yuey41 分钟前
【Linux】线程同步与互斥
linux·笔记
杨江42 分钟前
seafile docker安装说明
运维
舰长11544 分钟前
linux 实现文件共享的实现方式比较
linux·服务器·网络
好好沉淀1 小时前
Docker开发笔记(详解)
运维·docker·容器
zmjjdank1ng1 小时前
Linux 输出重定向
linux·运维
路由侠内网穿透.1 小时前
本地部署智能家居集成解决方案 ESPHome 并实现外部访问( Linux 版本)
linux·运维·服务器·网络协议·智能家居
树℡独1 小时前
ns-3仿真之应用层(三)
运维·服务器·ns3
VekiSon1 小时前
Linux内核驱动——基础概念与开发环境搭建
linux·运维·服务器·c语言·arm开发
zl_dfq2 小时前
Linux 之 【进程信号】(signal、kill、raise、abort、alarm、Core Dump核心转储机制)
linux