前言
有时我们需要更新同一行输出信息,而不是输出新行,如进度显示。两种实现方法:
- 输出退格符
\b
,清除该行输入字符 - 输出回车符
\r
,然后覆盖已输出的字符
对于第1种实现,该行已输出多少字符,需要刚好输出相同数量的\b
,以防止某些终端实现把上一行的回车也吃掉了。可使用printf()
的返回值得到已输出字符数。
对于第2种实现,输出的新字符数量必须不少于已输出的,以完全覆盖之前的输出。
本人更倾向于第2种实现,只要保证每行输出的字符数量相同(通过printf()
指定参数的输出字符数),实现更简洁、维护性更好。
本文分别给出C语言和Shell的实现例子。
C语言实现
c
#include <stdio.h> #include <unistd.h> int main(void) { int i; for (i = 0; i < 100; i++) { printf("\rprogress: %3d %%", i); fflush(stdout); sleep(1); } printf("\n"); return 0; }
需要特别注意的是,终端设备是行缓冲
设备,即除非输出换行符\n
或缓冲区已满,否则printf()
不会有任何输出;所以这里需要使用fflush(stdout)
清除标准输出缓冲区,以立即输出缓冲区内容。
Shell实现
bash
#!/bin/bash for i in {1..100} do printf "\rprocess %3d %%" $i sleep 1 done # print a new line echo
脚本中可使用printf
命令行程序。