etchar()和EOF总结

本文详细阐述了C语言中getchar()函数的使用技巧及EOF概念,通过实例代码和解释,帮助初学者理解字符输入过程中的细节。包括getchar()的读取方式、EOF的使用场景以及如何正确处理EOF带来的输入结束问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

大师级经典的著作,要字斟句酌的去读,去理解。以前在看K&R的The C Programming Language(SecondEdition)
第1.5节的字符输入/输出,被getchar()和EOF所迷惑了。可能主要还是由于没有搞清楚getchar()的工作原理和EOF的用法。因此,感觉很有必要总结一下,不然,很多琐碎的知识点长时间过后就会淡忘的,只有写下来才是最好的方法。


其实,getchar()最典型的程序也就几行代码而已。本人所用的环境是DebianGNU/Linux,在其他系统下也一样。
一、 getchar的两点总结:
1.getchar是以行为单位进行存取的。
当用getchar进行输入时,如果输入的第一个字符为有效字符(即输入是文件结束符EOF,Windows下为组合键Ctrl+Z, Unix/Linux下为组合键Ctrl+D),那么只有当最后一个输入字符为换行符'/n'(也可以是文件结束符EOF,EOF将在后面讨论)时, getchar才会停止执行,整个程序将会往下执行。譬如下面程序段:

while ( ( c = getchar ( ) ) ! = EOF ) { 
    putchar ( c) ; 
}


执行程序,输入:abc,然后回车。则程序就会去执行puchar(c),然后输出abc,这个地方不要忘了, 系统输出的还有一个回车(是因为我们输入abc之后又输入回车,而不是putchar自动添加) 。然后可以继续输入,再次遇到换行符的时候,程序又会把那一行的输入的字符输出在终端上。


对于getchar,肯定很多初学的朋友会问,getchar不是以字符为单位读取的吗?那么,既然我输入了第一个字符a,肯定满足while循环(c = getchar()) != EOF的条件阿,那么应该执行putchar(c)在终端输出一个字符a。不错,我在用getchar的时候也是一直这么想的,但是程序就偏偏不着样执行,而是必需读到一个换行符或者文件结束符EOF才进行一次输出。

对这个问题的一个解释是,在大师编写C的时候,当时并没有所谓终端输入的概念,所有的输入实际上都是按照文件进行读取的,文件中一般都是以行为单位的。因此,只有遇到换行符,那么程序会认为输入结束,然后采取执行程序的其他部分。同时,输入是按照文件的方式存取的,那么要结束一个文件的输入就需用到 EOF (Enf Of File) . 这也就是为什么getchar结束输入退出时要用EOF的原因。

2.getchar()的返回值一般情况下是字符,但也可能是负值,即返回EOF。

这里要强调的一点就是,getchar函数通常返回终端所输入的字符,这些字符系统中对应的ASCII值都是非负的。因此,很多时候,我们会写这样的两行代码:

char c; 
= getchar ( ) ;


这样就很有可能出现问题。因为getchar函数除了返回终端输入的字符外,在遇到Ctrl+D(Linux下)即文件结束符EOF时,getchar ()的返回EOF,这个EOF在函数库里一般定义为-1。因此,在这种情况下,getchar函数返回一个负值,把一个负值赋给一个char型的变量是不正确的。为了能够让所定义的变量能够包含getchar函数返回的所有可能的值,正确的定义方法如下(K&R C中特别提到了这个问题):

int c; 
= getchar ( ) ;

二、 EOF的两点总结(主要指普通终端中的EOF)
首先明确一下EOF的概念,EOF是在stdio.h中定义的一个常量,为#define EOF (-1)  用来表示文件的结尾,当某些函数读取到文件尾时便返回EOF。另外,不是说每个文件的尾部都有一个专门的标志用来标示文件结尾,更不是说每个文件尾都有EOF,我们可以想象一下,如果我们每次读取一个字符都要判断是否到达文件尾,那样效率也太低了,那如何判断文件是否已经结束了呢?中断、异常。当我们用函数读入文件数据的时候,函数总会返回一个状态,是读取成功还是失败,那么这个状态怎么表示呢,所以就约定俗成定义一个标识符表示这个状态,就有了EOF,BOF等等 

1.EOF作为文件结束符时的情况:

EOF虽然是文件结束符,但并不是在任何情况下输入Ctrl+D(Windows下Ctrl+Z)都能够实现文件结束的功能,只有在下列的条件下,才作为文件结束符。
(1)遇到 getcahr函数执行时,要输入第一个字符时就直接输入Ctrl+D,就可以跳出getchar(),去执行程序的其他部分;
(2)在前面输入的字符为换行符时,接着输入Ctrl+D;
(3)在前面有字符输入且不为换行符时,要连着输入两次Ctrl+D,这时第二次输入的Ctrl+D起到文件结束符的功能,至于第一次的Ctrl+D的作用将在下面介绍。
其实,这三种情况都可以总结为只有在getchar()提示新的一次输入时,直接输入Ctrl+D才相当于文件结束符。

2.EOF作为行结束符时的情况,这时候输入Ctrl+D并 不能结束getchar(),而只能引发getchar()提示下一轮的输入 。

这种情况主要是在进行getchar()新的一行输入时,当输入了若干字符(不能包含换行符)之后,直接输入Ctrl+D,此时的Ctrl+D并不是文件结束符,而只是相当于换行符的功能,即结束当前的输入。以上面的代码段为例,如果执行时输入abc,然后Ctrl+D,程序输出结果为:
abcabc 

注意:第一组abc为从终端输入的,然后输入Ctrl+D,就输出第二组abc,同时光标停在第二组字符的c后面,然后可以进行新一次的输入。这时如果再次输入Ctrl+D,则起到了文件结束符的作用,结束getchar()。
如果输入abc之后,然后回车,输入换行符的话,则终端显示为:
abc         //第一行,带回车 
abc         //第二行 
               //第三行 

其中第一行为终端输入,第二行为终端输出,光标停在了第三行处,等待新一次的终端输入。
从这里也可以看出Ctrl+D和换行符分别作为行结束符时,输出的不同结果。
EOF 的作用也可以总结为:当终端有字符输入时,Ctrl+D产生的EOF相当于结束本行的输入,将引起getchar()新一轮的输入;当终端没有字符输入或者可以说当getchar()读取新的一次输入时,输入Ctrl+D,此时产生的EOF相当于文件结束符,程序将结束getchar()的执行。
另外,有可能有些人会有疑问,既然没有专门的标志表示文件的结尾,那我们输入的Ctrl+D是什么?这是系统定义的,Linux系统定义了几个特殊组合键,其功能如下:
ctrl-c 发送 SIGINT 信号给前台进程组中的所有进程。常用于终止正在运行的程序。
ctrl-z 发送 SIGTSTP 信号给前台进程组中的所有进程,常用于挂起一个进程。
ctrl-d 不是发送信号,而是表示一个特殊的二进制值,表示 EOF。
ctrl-/ 发送 SIGQUIT 信号给前台进程组中的所有进程,终止前台进程并生成 core 文件。

【补充】本文第二部分中关于EOF的总结部分,适用于终端驱动处于一次一行的模式 下。也就是虽然getchar()和putchar()确实是按照每次一个字符 进行的。但是终端驱动处于一次一行的模式,它的输入只有到“/n”或者EOF时才结束,因此,终端上得到的输出也都是按行的。 
如果要实现终端在读一个字符就结束输入的话,下面的程序是一种实现的方法(参考《C专家编程》,略有改动): 

/*Edit by Godbach
  CU Blog: http://blog.chinaunix.net/u/33048/
*/
 
# include < stdio. h> 
# include < stdlib. h> 

int 
main( void ) 
{ 
    int c; 
    /* 终端驱动处于普通的一次一行模式 */ 
    system ( "stty raw" ) ; 
   
    /* 现在的终端驱动处于一次一个字符模式 */ 
    c = getchar ( ) ; 
    putchar ( c) ; 
   
    /* 终端驱动处又回到一次一行模式 */ 
     system ( "stty cooked" ) ; 
   
    return 0; 
}

编译运行该程序,则当输入一个字符时,直接输出一个字符,然后程序结束。
由此可见,由于终端驱动的模式不同,造成了getchar()输入结束的条件不一样。普通模式下需要回车或者EOF,而在一次一个字符的模式下,则输入一个字符之后就结束了。

希望本文可以对初学C的朋友提供一点帮助,也希望能和其他朋友进行交流。其中理解不对的地方若能得到指正和建议,本人将不胜感激。同时,本文参考了 chinaunix.net关于getchar讨论的帖子和一位博友的文章,链接地址分别为: 
http://blog.chinaunix.net/u/9861/showart_64652.html 
http://bbs.chinaunix.net/viewthread.php?tid=679688&extra=&page=1 
欢迎交流和指正。

 

 原文地址 http://blog.chinaunix.net/u/33048/showart_271120.html

### 回答1: 在shell脚本中,if语句用于根据条件执行不同的代码块。结合EOF(End Of File)可以方便地在脚本中插入多行文本。 具体用法如下: ``` if [ 条件判断 ]; then # 条件成立时执行的代码块 # 可以包含多行文本,使用EOF包裹 cat << EOF 多行文本1 多行文本2 ... EOF else # 条件不成立时执行的代码块 ... fi ``` 在上面的代码中,if语句根据条件判断是否执行多行文本。如果条件成立,则执行EOF包裹的多行文本,否则执行else部分的代码块。 需要注意的是,在EOFEOF之间的文本将作为一个整体,包括空格制表符。如果想要在多行文本中使用shell变量,需要将EOF改为'EOF',例如: ``` if [ 条件判断 ]; then # 条件成立时执行的代码块 # 可以包含多行文本,使用'EOF'包裹 cat << 'EOF' 多行文本1 $变量名 多行文本2 ... EOF else # 条件不成立时执行的代码块 ... fi ``` 在上面的代码中,$变量名不会被shell解析为变量,而是作为普通文本输出。 ### 回答2: shell中的if语句可以用于条件判断,而EOF(End Of File)是一个特殊的标记,用于指示输入的结束。在Shell脚本中,我们可以将if语句EOF结合使用来实现更复杂的逻辑控制。 使用ifEOF结合的常见用法是通过输入重定向来读取一个文件,并根据文件的内容执行相应的操作。具体的使用方法如下: ```shell if condition then command1 command2 #如果条件为真,则执行一系列命令 else command3 command4 #如果条件为假,则执行另一系列命令 fi << EOF file_content EOF ``` 上面的代码中,condition是if语句的判断条件,可以是一个表达式或者具体的值。如果条件为真,则执行then块中的命令,否则执行else块中的命令。 EOF后面是文件的内容,可以是一个文件名,也可以是具体的一些文本。通过输入重定向符号(<<),将文件内容作为输入传递给if语句。 在执行过程中,Shell会读取EOF之前的内容,并将其作为输入传递给if语句。根据实际的文件内容,if语句会执行相应的命令块。 这种使用方法可以将一组命令封装在一个文件中,然后通过ifEOF结合将文件内容作为输入传递给Shell脚本,从而实现更加复杂的逻辑控制。 需要注意的是,EOF可以是任意的标记,我们可以使用其他的标记代替,例如END、DATA等,只要保证开始结束的标记相同即可。 ### 回答3: shell中的if语句可以与EOF(End of File)结合使用来读取多行输入并进行条件判断。下面是一个示例: ```shell #!/bin/bash # 从用户输入中读取两个数字,并判断它们的大小关系 echo "请输入第一个数字:" read num1 echo "请输入第二个数字:" read num2 if [ $num1 -gt $num2 ]; then echo "第一个数字大于第二个数字" elif [ $num1 -lt $num2 ]; then echo "第一个数字小于第二个数字" else echo "两个数字相等" fi ``` 上面的脚本会提示用户依次输入两个数字,然后使用if语句判断它们的大小关系并输出相应的结果。使用EOF结合if语句的好处是可以方便地读取多行输入,而不需要用户重复输入。 ```shell #!/bin/bash # 读取文件内容并输出到标准输出 filename="example.txt" if [ -e $filename ]; then while IFS=' ' read -r line do echo "$line" done < $filename else echo "文件不存在" fi ``` 上面的脚本会判断文件是否存在,如果存在则使用EOF结合while语句读取文件的每一行内容,并使用echo语句输出到标准输出。如果文件不存在,则会输出相应的提示信息。 总结来说,shell中的if语句可以与EOF结合使用来读取多行输入并进行相应的条件判断处理。这样可以方便地操作文件或者处理用户的多行输入。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值