82

如何在 C 中擦除当前打印的控制台行?我正在使用 Linux 系统。例如 -

printf("hello");
printf("bye");

我想在同一行打印再见代替你好。

4

12 回答 12

82

您可以使用VT100 转义码。大多数终端,包括 xterm,都支持 VT100。对于擦除一条线,这是^[[2K. 在 C 中,这给出:

printf("\33[2K\r");
于 2009-10-02T09:33:30.510 回答
61

您可以使用\r(回车) 将光标返回到行首:

printf("hello");
printf("\rbye");

这将在同一行打印再见。但是它不会删除现有的字符,并且因为byehello短,所以你最终会得到byelo。要删除它,您可以使新打印的时间更长以覆盖多余的字符:

printf("hello");
printf("\rbye  ");

或者,首先用几个空格擦除它,然后打印你的新字符串:

printf("hello");
printf("\r          ");
printf("\rbye");

这将打印hello,然后转到行首并用空格覆盖它,然后再次回到开头并打印bye

于 2009-10-02T09:14:44.760 回答
56

一些有价值的微妙之处......

\33[2K擦除光标当前所在的整行

\033[A将光标向上移动一行,但在同一列中,即不在行首

\r将光标移至行首(r 用于回车 NB 回车不包括换行符,因此光标保持在同一行)但删除任何内容

特别是在 xterm 中,我尝试了上面提到的回复,我发现删除该行并从头开始重新开始的唯一方法是序列(来自@Stephan202 以及@vlp 和@mantal 发布的上述评论)\33[2K\r

'\n' 在实现说明中,为了让它在倒计时场景中正常工作,因为我没有在 each 的末尾使用换行符fprintf(),所以我每次都必须fflush()流(为了给你一些上下文,我开始xterm在Linux机器上使用fork而不重定向stdout,我只是fdfile用我坐在伪终端地址上的非阻塞文件描述符写入缓冲的FILE指针,在我的例子中是/dev/pts/21):

fprintf(fdfile, "\33[2K\rT minus %d seconds...", i);
fflush(fdfile);

请注意,我使用 \33[2K 序列来擦除行,然后使用\r回车序列将光标重新定位在行的开头。我不得不fflush()在每个之后,fprintf()因为我最后没有换行符'\n'。不需要 fflush() 的相同结果将需要额外的序列才能上一行:

fprintf(fdfile, "\033[A\33[2K\rT minus %d seconds...\n", i);

请注意,如果您在要写入的行的正上方有内容,它将被第一个 fprintf() 覆盖。您必须在上面多留一行,以允许第一次向上移动:

i = 3;
fprintf(fdfile, "\nText to keep\n");
fprintf(fdfile, "Text to erase****************************\n");
while(i > 0) { // 3 second countdown
    fprintf(fdfile, "\033[A\33[2KT\rT minus %d seconds...\n", i);
    i--;
    sleep(1);
}
于 2016-02-04T00:17:24.717 回答
4

您可以使用 \b 删除该行

printf("hello");
int i;
for (i=0; i<80; i++)
{
  printf("\b");
}
printf("bye");
于 2009-10-02T09:24:52.053 回答
3

通常,当字符串末尾有 '\r' 时,只打印回车而不打印任何换行符。如果您有以下情况:

printf("fooooo\r");
printf("bar");

输出将是:

barooo

我可以建议的一件事(也许是一种解决方法)是有一个以 NULL 结尾的固定大小字符串,该字符串初始化为所有空格字符,以 '\r' 结尾(每次打印前),然后使用 strcpy 将您的字符串复制到它(没有换行符),因此每次后续打印都将覆盖前一个字符串。像这样的东西:

char str[MAX_LENGTH];        
// init str to all spaces, NULL terminated with character as '\r'
strcpy(str, my_string);       // copy my_string into str
str[strlen(my_string)] = ' '; // erase null termination char
str[MAX_LENGTH - 1] = '\r';
printf(str);

您可以进行错误检查,以使my_string长度始终比 小一str,但您了解基本概念。

于 2009-10-02T09:22:47.577 回答
2

i遍历 char 数组单词。 j跟踪字长。 "\b \b"后退线时擦除单词。

#include<stdio.h>

int main()
{
    int i = 0, j = 0;

    char words[] = "Hello Bye";

    while(words[i]!='\0')
    {
        if(words[i] != ' ') {
            printf("%c", words[i]);
        fflush(stdout);
        }
        else {
            //system("ping -n 1 127.0.0.1>NUL");  //For Microsoft OS
            system("sleep 0.25");
            while(j-->0) {
                printf("\b \b");
            }
        }

        i++;
        j++;
    }

printf("\n");                   
return 0;
}
于 2012-02-17T23:41:23.127 回答
2

此脚本针对您的示例进行了硬编码。

#include <stdio.h>

int main ()
{

    //write some input  
    fputs("hello\n",stdout);

    //wait one second to change line above
    sleep(1);

    //remove line
    fputs("\033[A\033[2K",stdout);
    rewind(stdout);

    //write new line
    fputs("bye\n",stdout);

    return 0;
}

单击此处获取来源

于 2015-07-14T11:17:09.723 回答
1

您可以在这里使用一个简单的技巧,但是在打印之前需要进行准备,您必须将想要打印的内容放入变量中,然后打印,这样您就会知道删除字符串的长度。这是一个示例。

#include <iostream>
#include <string> //actually this thing is not nessasory in tdm-gcc

using namespace  std;

int main(){

//create string variable

string str="Starting count";

//loop for printing numbers

    for(int i =0;i<=50000;i++){

        //get previous string length and clear it from screen with backspace charactor

        cout << string(str.length(),'\b');

        //create string line

        str="Starting count " +to_string(i);

        //print the new line in same spot

        cout <<str ;
    }

}
于 2016-03-29T07:05:38.630 回答
0

刚刚找到这个旧线程,寻找某种转义序列来空白实际行。

很有趣的是,没有人想到 printf 返回写入的字符数(或者我错过了)。因此,只需打印 '\r' + 与 printf 返回的空白字符一样多,您将完全空白先前编写的文本。

int BlankBytes(int Bytes)
{
                char strBlankStr[16];

                sprintf(strBlankStr, "\r%%%is\r", Bytes);
                printf(strBlankStr,"");

                return 0;
}

int main(void)
{
                int iBytesWritten;
                double lfSomeDouble = 150.0;

                iBytesWritten = printf("test text %lf", lfSomeDouble);

                BlankBytes(iBytesWritten);

                return 0;
}

由于我无法使用 VT100,看来我必须坚持使用该解决方案

于 2018-02-01T18:00:27.903 回答
0
echo -e "hello\c" ;sleep 1 ; echo -e "\rbye  "

上面的命令会做什么:

  1. 它将打印你好,光标将保持在“o”(使用\c)

  2. 然后它将等待 1 秒(睡眠 1)

  3. 然后它将用再见替换你好。(使用\ r)

NOTE : Using ";", We can run multiple command in a single go.

于 2019-06-30T13:02:06.537 回答
0

在 Windows 10 下,可以通过在当前控制台中激活 VT100 模式来使用 VT100 样式,以使用如下转义序列:

#include <windows.h>
#include <iostream>

#define ENABLE_VIRTUAL_TERMINAL_PROCESSING 0x0004
#define DISABLE_NEWLINE_AUTO_RETURN  0x0008

int main(){

  // enabling VT100 style in current console
  DWORD l_mode;
  HANDLE hStdout = GetStdHandle(STD_OUTPUT_HANDLE);
  GetConsoleMode(hStdout,&l_mode)
  SetConsoleMode( hStdout, l_mode |
            ENABLE_VIRTUAL_TERMINAL_PROCESSING |
            DISABLE_NEWLINE_AUTO_RETURN );

  // create a waiting loop with changing text every seconds
  while(true) {
    // erase current line and go to line begining 
    std::cout << "\x1B[2K\r";
    std::cout << "wait a second .";
    Sleep(1);
    std::cout << "\x1B[2K\r";
    std::cout << "wait a second ..";
    Sleep(1);
    std::cout << "\x1B[2K\r";
    std::cout << "wait a second ...";
    Sleep(1);
    std::cout << "\x1B[2K\r";
    std::cout << "wait a second ....";
 }

}

请参阅以下链接:windows VT100

于 2019-08-19T15:49:29.040 回答
-1

其他人已经回答了OP的问题。对于那些想知道为什么回车在他们的 Linux 机器上的行为方式的人来说,这是一个答案 -

回车符的行为似乎与平台有关。

从 C11 标准的“5.2.2 字符显示语义”部分:

\r(回车)将活动位置移动到当前行的初始位置。

来自 POSIX 标准 ( https://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap03.html )的“3.86 回车符 (<carriage-return>)”部分:

输出流中的一个字符表示打印应该从发生回车的同一物理行的开头开始。它是C语言中'\r'指定的字符。未指定此字符是否是系统传输到输出设备以完成移动到行首的确切序列。

它没有说明回车是否应该删除(=> 填充 NUL 字符)整行。我的猜测是它不应该被删除。

但是,在我的 Linux 机器上(在 x86_64 和 ARM32 上都试过),我观察到回车符将光标移动到当前行的开头,并且还用 '\0' 字符(NUL 字符)填充了该行。为了注意那些 NUL 字符,您可能必须直接从您的代码中调用 write 系统调用,而不是通过 glibc printf 调用。

我们以下面的代码片段为例:

printf("hello");
printf("\rbye");

在 beaglebone black(32 位 ARM)bash 终端上构建和运行它:

ubuntu@arm:~$ ./a.out 
byeubuntu@arm:~$ 

写入系统调用时的 strace 输出:

bye)               = 9 9hello
+++ exited with 4 +++
于 2021-07-08T02:18:03.827 回答