一、进度条原理
进度条的的动态增长是利用人的视觉短暂停留效果的,不断从输出缓冲区刷新出相同的内容,在肉眼看来进度条在不断的增长。
1、显示问题:
因为不需要多行显示,只需刷新当前进度 ,那么输出时只需要回车‘\r’,不需要换行‘\n’
2、关于缓冲区:
linux下的每一个进程会维护一个print/scanf的缓冲区,对于缓冲区而言,缓冲区有一个概念叫做缓冲方式,就是说达到一定的方式,缓冲区的内容才会被刷新。
标C库函数自带缓冲区 ,常见的三种缓冲方式:
(1)行缓冲:遇到”\n”,就会刷新缓冲区
(2)全缓冲:把缓冲区写满,再进行刷新缓冲区
(3)无缓冲:系统调用函数无缓冲区(如:write)
默认情况下,一般采用行缓冲模式
当程序退出缓冲区自动刷新,也就是一个进程结束的时候,当使用scanf也会刷新。
所以,为了每次进行打印每行的内容,我们需要采用强制刷新缓冲区,这里有一个函数fflush()。
二、进度条的两种实现
1、用C语言实现
这里采用for循环打印的方式实现:
(1)程序代码:
1 #include<stdio.h>
2 #include<stdlib.h>
3 #include<unistd.h>
4
5 int main()
6 {
7 char buf[102]={0};//因为1—100是101个数,最后还要存一个‘\0’,所以大 小为102
8 char *str="-\|/";
9 int i=0;
10 for(;i<=100;++i)
11 {
12 printf("\033[034m[%-100s],[%d%%],[%c]\r",buf,i,str[i%4]);
13
14 fflush(stdout);
15 usleep(20000);
16 buf[i]='#';
17 }
18 printf("\n");
19 return 0;
20 }
运行结果:
2、用shell脚本语言实现彩色进度条
在用shell编写彩色进度条的时候,我们需要了解:
设定printf在终端输出字体的颜色:
终端字符的颜色是用转义序列进行控制,是文本模式下的系统显示功能,和具体的语言无关。
转义序列是以 ESC 开头,可以用 \033 完成相同的工作(ESC 的 ASCII 码用十进制表示就是 27, = 用八进制表示的 33)。
- 格式是:\033[显示方式;前景色;背景色m
类别 | 颜色 |
---|---|
显示方式 | 0(默认值)、1(高亮)、22(非粗体)、4(下划线)、24(非下划线)、5(闪烁)、25(非闪烁)、7(反显)、27(非反显) |
前景色 | 30(黑色)、31(红色)、32(绿色)、 33(黄色)、34(蓝色)、35(洋红)、36(青色)、37(白色) |
背景色 | 40(黑色)、41(红色)、42(绿色)、 43(黄色)、44(蓝色)、45(洋红)、46(青色)、47(白色) |
代码实现:
实现简单的彩色进度条:
1 #!/bin/bash
2
3 i=0
4 b=""
5 array=('-' '\\' '|' '/')
6
7 while [ $i -le 100 ]
8 do
9 let idx=i%4
10 printf "\e[31m\033[40m[%-100s]\e[32m\033[47m [%d%%] \e[30m \033[47m [%c] \e[0m\r" "$b" "$i" "${array[$idx]}"
11 b+='#'
12 usleep 200000
13 let i++
14 done
运行结果:
改进版:
1 #!/bin/bash
2
3 i=0
4 str=""
5 array=("\\" "|" "/" "-")
6 while [ $i -le 100 ]
7 do
8 let idx=i%4
9 if [ $i -le 20 ]
10 then
11 let color=44
12 let bg=34
13 elif [ $i -le 50 ]
14 then
15 let color=43
16 let bg=31
17 else
18 let color=42
19 let bg=33
20
21 fi
22 printf "\033[${color};${bg}m%-s\033[0m [%d%%] [%c]\r" "$str" "$i" "${array[$idx]}"
23
24 usleep 300000
25 let i++
26 str+="#"
27 done
28 printf "\n"
运行结果:
这个进度条在不同的区间颜色不尽相同。