原地更新终端输出信息

649 查看

前言

有时我们需要更新同一行输出信息,而不是输出新行,如进度显示。两种实现方法:

  1. 输出退格符\b,清除该行输入字符
  2. 输出回车符\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命令行程序。