0

我正在编写臭名昭著的书“Prentice Hall Software Series”,并尝试他们编写的代码并对其进行修改以了解更多关于 C 的信息。

我正在控制台中的 Fedora 25 上使用 VIM。以下代码是书中的引用,我知道缺少“int”以及 argc 和 argv 等。

Kernighan 和 Ritchie - C 编程语言:第 20 页

#include <stdio.h>
/* copy input to output; 1st version  */   
main(){       
    int c;
    c = getchar();       
    while (c != EOF) {           
        putchar(c);           
        c = getchar();       
    }   
}

使用此代码,我无法使“EOF”正常工作。我不确定“ctr + z”是否真的是真正要做的事情,因为它会退出控制台中的任何控制台程序。

好吧,因为我不确定我将条件更改为

...
while (c != 'a') {
...

所以通常如果我输入“a”,while 条件应该会中断并且程序应该终止。好吧,当我尝试运行它并输入'a'时它不会。这里有什么问题?

感谢你们!

4

1 回答 1

2

代码没有任何问题(除了 的古老声明main)。

通常,在 Unix 上,文件的结尾由 ctrl-D 向程序发出信号。如果您直接按 ctrl-D 或按换行符后按,您的程序将读取EOF.

然而,上面的简短解释隐藏了很多微妙之处。

在 Unix 中,终端输入可以以两种模式之一运行(称为 IIRC 生和熟)。在熟模式(默认)下,操作系统将缓冲来自终端的输入,直到它读取新行或 ctrl-D 字符。然后它将缓冲的输入发送到您的程序。

最终,您的程序将使用read系统调用来读取输入。read将返回读取的字符数,但通常会阻塞,直到它有一些字符要读取。getchar然后将它们一一传递给它的调用者。因此getchar,在处理该行中的任何字符之前,将阻塞直到接收到整行文本。(这就是为什么当你使用它时它仍然不起作用的原因a)。

按照惯例,read系统调用在文件结束时返回 0。这是如何getchar知道返回EOF给调用者的。ctrl-D 具有强制读取和清空缓冲区的效果(如果它在新行之后立即发送),即使没有人关闭输入流read,它看起来也像是到达 EOF。getchar这就是为什么 ctrl-D 在换行后直接按下时起作用,但在输入某些字符后按下时不起作用的原因。

于 2017-03-14T10:14:22.000 回答