C-为什么SCANF可以读取1024个字符,而STDIN流缓冲区仅为1024字节



我可以通过 apue 的代码在Ubuntu下连接我的终端的stdin流缓冲区大小和缓冲区类型:

#include "apue.h" //It's merely a wrapper header
void    pr_stdio(const char *, FILE *);
int     is_unbuffered(FILE *);
int     is_linebuffered(FILE *);
int     buffer_size(FILE *);
int
main(void)
{
    FILE    *fp;
    fputs("enter any charactern", stdout);
    if (getchar() == EOF)
        err_sys("getchar error");
    fputs("one line to standard errorn", stderr);
    pr_stdio("stdin",  stdin);
    pr_stdio("stdout", stdout);
    pr_stdio("stderr", stderr);
    if ((fp = fopen("/etc/passwd", "r")) == NULL)
        err_sys("fopen error");
    if (getc(fp) == EOF)
        err_sys("getc error");
    pr_stdio("/etc/passwd", fp);
    exit(0);
}
void
pr_stdio(const char *name, FILE *fp)
{
    printf("stream = %s, ", name);
    if (is_unbuffered(fp))
        printf("unbuffered");
    else if (is_linebuffered(fp))
        printf("line buffered");
    else /* if neither of above */
        printf("fully buffered");
    printf(", buffer size = %dn", buffer_size(fp));
}
int
is_unbuffered(FILE *fp)
{
    return(fp->_flags & _IO_UNBUFFERED);
}
int
is_linebuffered(FILE *fp)
{
    return(fp->_flags & _IO_LINE_BUF);
}
int
buffer_size(FILE *fp)
{
    return(fp->_IO_buf_end - fp->_IO_buf_base);
}

从终端中运行上述代码: stream = stdin, line buffered, buffer size = 1024


然后我写一个测试:

#include "stdio.h"
int main(){
    char c[2048];
    c[1033] = 'a';
    scanf("%s", c); //I paste 1440 '1' into the terminal.
    printf("%c", c[1033]); //I expect it to be 'a'.
    return 0;
}

i粘贴1440(> 1024个字节)字符'1'到终端,并期望以某种方式丢弃过多的输入数据,因为线缓冲区大小仅为1024字节。但是最终,我用c[1033]打印了" 1"。

为什么scanf可以读取1024个字符,而stdin的缓冲区大小仅为1024b?

引用 apue

我们可以看到,该系统的默认值是具有标准输入和标准 输出线连接到终端时被缓冲。线缓冲区为1,024 字节。请注意,这并不限制我们到1,024字节输入和输出线;只是 缓冲区的大小。编写2,048字节为标准输出将需要两个 编写系统调用。

或者我应该问如何?

我不太了解强调的文本。缓冲区不限制用户输入大小吗?为什么它说我可以输入比缓冲区大小数据更多的输入(对于每个输入)?有人可以告诉我它如何工作?

如注释中所述,当scanf()到达第一个缓冲区的末尾时,如果仍然需要更多数据,它将返回到系统以获取更多,可能很多次。缓冲区只是一种便利和优化度量。

最新更新