Почему scanf может читать более 1024 символов, в то время как буфер потока stdin составляет всего 1024 байта? - PullRequest
2 голосов
/ 06 марта 2019

Я могу получить stdin размер буфера потока и тип буфера, который подключал мой терминал под Ubuntu, по коду из APUE :

#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 character\n", stdout);
    if (getchar() == EOF)
        err_sys("getchar error");
    fputs("one line to standard error\n", 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 = %d\n", 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;
}

Я вставляю символ 1440 (> 1024 байта) в терминал и ожидаю, что лишние входные данные будут как-то отброшены, поскольку размер буфера строки составляет всего 1024 байта. Но в конце концов я напечатал «1» с c[1033].

Почему scanf может читать более 1024 символов, в то время как размер буфера stdin равен только 1024B?

Цитируется по APUE :

Мы видим, что по умолчанию для этой системы используется стандартный ввод и стандарт выходная линия буферизируется, когда они подключены к терминалу. Строка буфера составляет 1024 байт. Обратите внимание, что это не ограничивает нас входными и выходными строками по 1024 байта; это просто размер буфера. Для записи строки размером 2048 байт в стандартный вывод потребуется два записывать системные вызовы.

Или, может быть, я должен спросить, как?

Я не очень хорошо понимаю выделенный текст. Разве буфер не ограничивает размер пользовательского ввода? Почему написано, что я могу ввести больше данных о размере буфера ( для каждого ввода )? Может кто-нибудь сказать мне, как это работает?

1 Ответ

3 голосов
/ 06 марта 2019

Как отмечается в комментарии, когда scanf() достигает конца первого заполненного буфера, если ему все еще нужно больше данных, он возвращается в систему, чтобы получить больше, возможно, много раз.Буфер - это просто мера удобства и оптимизации.

...