在C中使用EOF(Ctrl + D)在getchar()中

vc6uscn9  于 2023-08-03  发布在  其他
关注(0)|答案(1)|浏览(75)

我正在看K&R中的1.9节,我不明白下面实现的代码中EOF的用法

#include <stdio.h>
#define MAXLINE 1000            /* maximum input line length */

int mymygetline(char line[], int maxline);
void copy(char to[], char from[]);

/* print the longest input line */
int main()
{
    int len;                      /* current line length */
    int max;                      /* maximum length seen so far */
    char line[MAXLINE];           /* current input line */
    char longest[MAXLINE];        /* longest line saved here */
    max = 0;

    while ((len = mymygetline(line, MAXLINE)) > 0) {
        if (len > max) {
            max = len;
            copy(longest, line);
        }
    }
    if (max > 0)                  /* there was a line */
        printf("%s", longest);
    return 0;
}

/* mymygetline: read a line into s, return length */
int mymygetline(char s[], int lim)
{
    int c, i;
    for (i = 0; i < lim - 1 && (c = getchar()) != EOF && c != '\n'; ++i)
        s[i] = c;
    if (c == '\n') {
        s[i] = c;
        ++i;
    }
    s[i] = '\0';
    return i;
}

/* copy: copy 'from' into 'to'; assume to is big enough */
void copy(char to[], char from[])
{
    int i;
    i = 0;
    while ((to[i] = from[i]) != '\0')
        ++i;
}

字符串
我尝试的是:

  • 初始程序运行良好,输入标记为“\n”的差异行易于理解
  • 当我第一次尝试输入EOF时,它只是结束最近的输入流,并将getchar()设置为等待输入流的模式。

问:

  • 当我输入EOF两次内联,第一个EOF只是像一个“\n”或其他一些机制,或者我只是错过了一些东西在这里。
qco9c6ql

qco9c6ql1#

以下是按键如何传播到代码(mymygetline())函数:

  • 用户按下键。
  • 键盘接收按键并向类Unix操作系统发送某种包含键码的事件。
  • 操作系统计算出事件应该被路由到终端(这包括GUI上的终端模拟器窗口)。操作系统将事件转换为字节序列并将其发送到终端。
  • 终端可以进行一些更多的变换和缓冲。默认情况下,它会缓冲(即不允许通过)字节,直到接收到换行符或特殊字符(例如与Ctrl-D 对应的字符)。
  • 程序中的libc代码(由getchar(3)调用)从终端读取一个字节序列。当终端缓冲时,libc代码阻塞read(2)系统调用。只要终端刷新,read(2)就会一次接收许多字节,通常是一整行,后跟一个换行符或一个EOF指示符(read(2)返回0)。
  • 您的代码(mymygetline())从getchar(3)接收read(2)读取的字节,每次一个字节。一旦没有更多的字节可用,libc代码就会再次调用read(2),它会阻塞直到有更多的字节到来。

如果用户按下Ctrl-D,read(2)返回到目前为止缓冲的字节(如果有的话),随后的read(2)(由第二个Ctrl-D 触发)返回0,这导致getchar(3)返回EOF。

相关问题