🕺作者: 主页
我的专栏 C语言从0到1 探秘C++ 数据结构从0到1 探秘Linux 菜鸟刷题集 😘欢迎关注:👍点赞🙌收藏✍️留言
🏇码字不易,你的👍点赞🙌收藏❤️关注对我真的很重要,有问题可在评论区提出,感谢阅读!!!
前言
前面几章我们讲述了Linux指令、权限管理、编辑器、编译器、软件包管理器、自动化管理工具,也写了简单的”Hello World“程序,但是没有实现过复杂一点的程序,光说不练假把式,今天就带大家来做一个小项目–进度条。
搭建框架
在写代码之前,我们先把文件先建好。
vim main.c
vim process.c
vim process.h
vim makefile
书写代码
在写代码要先讲一个知识点。
1.回车换行
在我们日常使用回车键时,是否注意到,当我们按下回车键,也就是\n
,光标就会换到第二行的开始,其实这里面有两个动作,换行和把光标设置到行开始,在C语言中可用\r
来把光标换到开始位置
我们来借此写个小demo:倒计时void download() { int cnt=10; while(cnt!=0) { printf("%-2d\r",cnt); fflush(stdout); cnt--; sleep(1); } printf("\n"); }
效果如下:
第一版
我们最终要实现这个效果:
我们前面讲了\r
可以把光标移到行始,所以在我们输出缓存字符的时候,每一次都移到一开始的位置,随着缓存字符的增多,来实现进度条的效果。
代码如下:
process.c
#include"process.h"
const char *lable="|/-\\";//通过循环实现一个转动的效果
char bar[NUM];
void download(int speed)
{
char bar[NUM];//使用数组来存储”缓存字符“
memset(bar,'\0',sizeof(bar));//初始化
int len =strlen(lable);
int cnt=0;
while(cnt<=TOP)//设置终止条件
{
printf("[%-100s][%d%%][%c]\r", bar, cnt, lable[cnt%len]);
//%-100s:预留100个空间左对齐
//%%:输出%,也可以使用\%
//\r:把光标移到开始位置
fflush(stdout);//清空缓存区
bar[cnt++]=BODY;//将cnt所在位置设置为”BODY"
if(cnt<100)bar[cnt]=RIGHT;//设置>
usleep(speed);//控制进度条的速度,单位是微秒
}
printf("\n");
}
main.c
#include "process.h"
int main()
{
download(100000);
return 0;
}
process.h
#pragma once
#include<stdio.h>
#include <unistd.h>
#include<string.h>
#define NUM 102
#define TOP 100
#define BODY '='
#define RIGHT '>'
extern void download(int speed);
makefile
process:process.c main.c
gcc -o $@ $^
.PHONY:clean
clean:
rm -f process
通过注释,应该可以清楚明白代码的实现原理,但是进度条该怎么用呢?这是一个问题,下面我们来模拟一下使用场景,只需要再修改一下代码即可。
第二版
参考代码注释理解
main.c
#include "processBar.h"
typedef void (*callback_t)(int); // 函数指针类型
// 模拟一种安装或者下载
void downLoad(callback_t cb)
{
int total = 1000; // 1000MB
int curr = 0; // 0MB
while(curr <= total)
{
// 模拟进行着某种下载的任务, 我
usleep(50000); // 模拟下载花费的时间
int rate = curr*100/total; // 更新进度
cb(rate); // 通过回调,展示进度
curr += 10; // 循环下载了一部分
}
printf("\n");
}
int main()
{
printf("donwnload 1: \n");
downLoad(processbar);
initbar();
printf("donwnload 2: \n");
downLoad(processbar);
initbar();
printf("donwnload 3: \n");
downLoad(processbar);
initbar();
printf("donwnload 4: \n");
downLoad(processbar);
initbar();
return 0;
}
processBar.h
#pragma once
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <unistd.h>
#define NUM 102
#define TOP 100
#define BODY '='
#define RIGHT '>'
extern void processbar(int rate);
extern void initbar();
processBar.c
#include "processBar.h"
const char *lable="|/-\\";
char bar[NUM];
// 是如何被调用的
void processbar(int rate)
{
if(rate < 0 || rate > 100) return;
int len = strlen(lable);
printf("[%-100s]""[%d%%][%c]\r", bar, rate, lable[rate%len]); // 没有\n,就没有立即刷新,因为显示器模式是行刷新
fflush(stdout);
bar[rate++] = BODY;
if(rate < 100) bar[rate] = RIGHT;
}
void initbar()
{
memset(bar, '\0', sizeof(bar));
}
makefile
processbar:processBar.c main.c
gcc -o $@ $^
.PHONY:clean
clean:
rm -f processbar
效果是一样的,看到这里,我相信你对实现一个进度条已经非常了解了,但是有没有觉得它有点不好看,那么我们再来优化一下。
第三版
怎么修改?这里要了解一下怎么让编译器输出颜色,可参考这篇文章
http://t.csdn.cn/O2T8F
只需修改processBar.c
文件即可
如下:
#include "processBar.h"
const char *lable="|/-\\";
char bar[NUM];
// 是如何被调用的
void processbar(int rate)
{
if(rate < 0 || rate > 100) return;
int len = strlen(lable);
printf("\033[38;2;128;0;128m\033[48;2;255;255;255m[%-100s][%d%%][%c]\033[m\r", bar, rate, lable[rate % len]);
fflush(stdout);
bar[rate++] = BODY;
if(rate < 100) bar[rate] = RIGHT;
}
void initbar()
{
memset(bar, '\0', sizeof(bar));
}
效果如下:
后记
本篇我们讲述了如何在Linux上实现一个进度条,并对其进行了美化,但是我们的目的是通过做项目来把之前所学的相关知识串起来,我并没有对它进行过多的讲解,因为它本身并不难,难的是你在linux系统上编程的过程,但它也是最重要的!