文章目录
一、理解回车换行
在我们熟悉的C语言中,换行就可以跳转的下一行开头 ,但其实这一操作有两个步骤,\r (回车)和 \n(换行)
\r 回车就是回到这一行开头
\n 换行就是另起一行
二、认识行缓冲
在内存中预留了一块空间,用来缓冲输入或输出的数据,这个保留的空间被称为缓冲区。下面我们通过几个代码例子来理解行缓冲概念
1、代码一、二(回车换行理解)
代码一:
1#include <stdio.h>
2 int main()
3 {
4 printf("hello world\n");
5 return 0;
6 }
代码二:
1#include <stdio.h>
2 int main()
3 {
4 printf("hello world\r");
5 return 0;
6 }
不难发现 \n 可以打印出来,而 \r,不能打印出来,因为显示器模式是行刷新缓冲区是按行缓冲的,没有\n,就不能立即刷新。 \r 回到行首后,会进行覆盖写,shell 提示符会覆盖掉之前写的 “hello world”,如果我们在 “hello world” 不加 \r,则不会进行覆盖写,shell 提示符会顺着 “hello world” 往后写
如下:
2、代码三、四(sleep函数和ffush函数理解)
行缓冲是缓冲区刷新策略的一种,在行缓冲模式下,当输入和输出中遇到 ‘\n’ 换行时,就会刷新缓冲区,下面我们认识头文件<unistd.h>的三个函数
s l e e p sleep sleep : Linux 下的休眠函数,单位是秒
u s l e e p usleep usleep:和sleep 一样,单位ms(即10-6 m)
f f l u s h fflush fflush :刷新缓冲区
代码 3:
1 #include <stdio.h>
2 #include <unistd.h>
3 int main()
4 {
5 printf("hello world");
6 sleep(3);
7 return 0;
8 }
我们知道函数代码语句是从上到下依次进行的,而我们看到的却是先休眠三秒,然后再打印出"hello world",原因是因为数据保存在缓冲区中,没有主动刷新。当程序退出后,保存在缓冲区中的数据被自动刷新出来了,如果我们想提前刷新,便可以调用 f f l u s h fflush fflush函数来刷新缓冲区
代码四:
1 #include <stdio.h>
2 #include <unistd.h>
3 int main()
4 {
5 printf("hello world");
6 fflush(stdout);
7 printf("\n");
8 sleep(3);
9 return 0;
10}
这次 “hello world” 被直接打印出来,我们加 \n避免 s h e l l shell shell 提示符出现在 “hello world” 后面
三、简单倒计时
1、效果展示
有了以上的准备工作,能够让我们理解倒计时和进度条工作原理,下面我们先给代码和效果,在讲其中细节2、倒计时代码
1 #include <stdio.h>
2 #include <unistd.h>
3 int main()
4 {
5 int cnt=10;
6 while(cnt>=0)
7 {
8 printf("%-2d\r",cnt);
9 fflush(stdout);
10 sleep(1);
11 cnt--;
12 }
13 printf("\n");
14 return 0;
15 }
3、实现过程分析
1、定义倒计时变量 cnt,让其逐渐递降
2、核心就是用 \r 回到缓冲区行首进行覆盖写,然后fflush不断刷新出出来
3、格式调整,打印 cnt==10 时,在缓冲区打印的其实是字符1和字符0,如果我们不用 2d% 来调整格式,而用 d% 的话,那么覆盖写只会覆盖第一位字符 1 1 1 的位置,而第二位的字符 0 0 0, 还留在缓冲区原来的位置,于是倒计时便会变为下面这样
10->90->80->70->60->50->40->30->20->10->00 ,-2d% 加个负号保证其向左对齐
4、倒计时完加个 \n符,shell 提示符就不会出现在倒计时后面
四、进度条
1、效果展示
2、进度条代码
makefile
1 processbar:processBar.c main.c
2 gcc -o $@ $^
3 .PHONY:clean
4 clean:
5 rm -f processbar
头文件processBar.h
1 #pragma once
2 #include<stdio.h>
3
4 #define SIZE 102
5 #define BODY '='
6 #define TAIL '>'
7
8 extern void processbar(int rate);
9 extern void initbar();
.c 文件processBar.c
1 #include "processBar.h"
2 #include <string.h>
3 #include <unistd.h>
4
5 #define LIGHT_GREEN "\033[1;32m" //亮绿色
6 #define NONE "\033[m" //截断
7
8 const char *lable="|/-\\";
9 char bar[SIZE];
10
11 void processbar(int rate)
12 {
13 if(rate<0 || rate > 100)
14 return;
15 //没有\n,就没有立即刷新,因为显示器模式是行刷新
16 printf(LIGHT_GREEN"[%-100s]"NONE"[%d%%][%c]\r",bar,rate,lable[rate%4]);
17 fflush(stdout);
18 bar[rate++]=BODY;
19 if(rate<100)
20 bar[rate]=TAIL;
21 }
22
23 void initbar()
24 {
25 memset(bar,'\0',sizeof(bar));
26 }
.c文件main.c
1 #include "processBar.h"
2 #include <unistd.h>
3
4 typedef void(*callback_t)(int);
5 void downLoad(callback_t cb)
6 {
7 initbar();
8 int total=1000; // 1000MB
9 int curr=0; //0MB
10 while(curr<=total)
11 {
12 // 进行某种下载任务
13 usleep(50000); //模拟下载花费时间
14 int rate=curr*100/total;//更新进度
15
16 cb(rate); //回调展示进度
17 //processbar(curr*100/total);
18
19 curr+=10; // 循环下载了一部分,更新进度
20 }
21 printf("\n");
22 }
23
24
25 int main()
26 {
27 printf("download 1:\n");
28 downLoad(processbar);
29 printf("download 2:\n");
30 downLoad(processbar);
31 printf("download 3:\n");
32 downLoad(processbar);
33 printf("download 4:\n");
34 downLoad(processbar);
35 return 0;
36 }
3、实现过程分析
进度条实现样式
进度条样式 :
主体样式为两个中括号包裹,中间 => 推进的方式呈现,比如:[======>]
主体右侧中括号位置保持不变,中间元素不断推进,比如:[=> ]
因此我们把中间主体 = 宏定义为 BODY ,把尾侧 > 宏定义为 TAIL
进度条百分比:
显示当前加载进度,用 [rate%] 显示,rate 随着进度条的不断推进而变化,而打印 %(转义字符)则需要两个 %%
进度条旋转字符:
显示加载样式,可以利用一个旋转的字符,例如 [] 的样式,顺时针不断旋转,依次为 “| / - \”,注意 \ 也是转义字符,因此需要两个 \ \,对此我们定义一个lable指针指向常量字符串const char *lable=" | / - \ "
进度条颜色: c语言颜色参考
我们可以根据自己的喜好给进度条上色,在此我们找到颜色参照表
把亮绿色宏定义为 #define LIGHT_GREEN “\033[1;32m”
结束的地方宏定义为 #define NONE “\033[m”
进度条实现方法
预留进度条大小为 100 个 = ,外加 1 个 > ,加上保存 \0 的位置,建一个存储为102 (宏定义为SIZE)个单位的bar数组。processbar()函数,将某比率的进度条打印出来,\r 回到行首后,fflush 刷新缓冲区,同时bar数组尾插上 =,再插入 >,注意当rate为100时,进度条加载完毕,因此不需要再插入 >
将其模拟为执行某个下载任务,把实现方法 processbar()函数 作为参数传递给 downLoad()模拟下载函数 ,对此我们定义一个processbar()类型的函数指针,重命名为 callback_t。我们定义下载包大小为
t
o
t
a
l
total
total ,当前进度为
c
u
r
r
curr
curr ,因此进度比例为 rate=curr*100/total。用usleep函数模拟下载时间,然后循环起来回调processbar()函数,便实现了进度条,最后考虑到第二次下载,bar数组满了,我们再每次调用downLoad()函数时,清空bar数组