int main(void){
char cmdline[MAXLINE];
while(1){
printf("> ");
fgets(cmdline, MAXLINE, stdin);
if(feof(stdin)){
exit(0);
}
eval(cmdline);
}
}这是教授给我的myShell计划的主要部分。
但有一件事我在代码里不明白。
这里写着如果(feof(Stdin))出口(0);
标准输入的结尾是什么?
fgets接受所有字符,直到输入enter键。典型的“文件”(e.g.txt)的结尾在直觉上是可以理解的,但是标准输入的结束意味着什么呢?
在什么实际情况下,feof(stdin)实际上返回true?
即使您输入一个空格而没有输入任何内容,if语句也不会传递。
发布于 2022-03-24 12:36:44
feof测试流的文件结束指示符,并在文件结束指示符设置的情况下返回true (非零)。
对于常规文件,尝试在文件结束后读取设置文件结束指示符.对于终端,一种典型的行为是,当程序试图从终端读取而没有获得数据时,文件结束指示符就会被设置。在具有默认设置的Unix系统中,触发这种“无数据、文件结束行为”的方法是在行的开头或在先前的控件-D之后立即按下control。
之所以这样做,是因为control的意思是“立即向程序发送挂起的数据”。这将被进一步描述为in this answer。
因此,如果要结束程序的输入,请按control(如果不是在行的开头,则按第二次)。
对于来自终端的输入,虽然这确实会导致文件结束指示,但它实际上并没有结束输入或关闭流。该程序可以清除文件末尾的指示符,并继续阅读.即使是常规文件,程序也可以清除文件结束指示符,将文件上下文重置到不同的位置,并继续读取。
发布于 2022-03-24 12:04:01
混淆的是假定stdin =终端。这不一定是真的。
什么是stdin取决于您如何运行您的程序。例如,假设您的可执行文件名为a.out,如果您像这样运行它:
echo "foo" | ./a.outStdin是一个不同进程的输出,在这个例子中,这个过程只是输出单词"foo",因此stdin将包含"foo“,然后是EOF。
另一个例子是:
./a.out < file.txt在这种情况下,stdin是"file.txt“。当文件被读取到末尾时,stdin将得到EOF。
Stdin也可以是一种特殊的设备,例如:
./a.out < /dev/random在这种特殊情况下,它是无限的。
最后,当您简单地运行您的程序和stdin是终端-您也可以生成EOF -只需按CTRL,这会发送一个特殊的符号意味着EOF到终端。
P.S.还有执行进程的其他方法。在这里,我只给出了从命令行shell执行的进程的例子。但是进程可以由不同的进程执行,而不一定是从shell执行的。在这种情况下,进程的创建者可以决定什么是stdin -终端、管道、套接字、文件或任何其他对象。
https://stackoverflow.com/questions/71601638
复制相似问题