1

总之,我正在设计一个Key-Value服务器,当我写客户端的时候,我发现了一个很奇怪的事情,看简化代码:

while(1)
{
    printf("->:");
    read(STDIN_FILENO, buf, sizeof(buf));
    write(client_sock, buf, sizeof(buf));
    int m = read(client_sock, buf, sizeof(buf));
    buf[m] = '\0';
    printf("%s", buf);
}

当我运行程序时,它首先要求输入,所以我输入了一些东西,但什么也没发生!(当我使用其他客户端时,服务器运行良好,并且很好地回显了一些东西)

然后我只更改了一行代码:

printf("\n->:");

然后它运行良好!为什么?为什么“\n”可以改变输出?我想可能是 read() ,但我无法解释

4

2 回答 2

5

printf(3)是 C 标准 IO 库的一部分,它执行内部缓冲以提高性能。

缓冲分为三种类型:无、行和块。

应用哪个缓冲部分取决于要写入的描述符是否存在2,以及它是否连接到终端。(见isatty(3)。)

如果打印到 stderr ( 2) 则不进行缓冲。

如果对任何其他描述符进行了打印,则无论它是否为终端,行为都会发生变化:如果输出是终端,则输出为line buffered。如果输出不是终端(文件、管道、套接字等),则输出是块缓冲的。

\n当行缓冲时,它会在打印任何内容之前等待。(或者,如果您在发送之前写入的内容足以溢出内部缓冲区\n。)

我推荐的是以下内容:

printf("->:");
fflush(stdout);
read(STDIN_FILENO, buf, sizeof(buf));
/* ... */
printf("%s\n", buf);

这是一个很小的变化;你不会在程序开始时得到一个毫无意义的空行,并且提示应该立即显示出来。

您可以使用该setvbuf(3)函数在启动时更改一次流的缓冲,如果您愿意,则无​​需再次刷新它。

int err = setvbuf(stdout, NULL, _IONBF, 0);
/* check err */
于 2011-12-18T06:13:43.530 回答
3

默认情况下,标准输出是行缓冲的。如果您不写完整的行,则输出将保存在缓冲区中,直到您写完为止。您可以使用fflush刷新流或setbuf更改缓冲模式。

于 2011-12-18T06:12:27.943 回答