C程序設計:getchar和EOF問題

轉載:http://bbs.chinaunix.net/thread-981231-1-1.html
大師級經典的著作,要字斟句酌的去讀,去理解秘血。以前在看K&R的The C Programming Language(SecondEdition)第1.5節(jié)的字符輸入/輸出肝陪,被getchar()和EOF所迷惑了〖娣幔可能主要還是由于沒有搞清楚getchar()的工作原理和EOF的用法。因此,感覺很有必要總結一下唆缴,不然鳍征,很多瑣碎的知識點長時間過后就會淡忘的,只有寫下來才是最好的方法面徽。其實艳丛,getchar()最典型的程序也就幾行代碼而已。本人所用的環(huán)境是DebianGNU/Linux趟紊,在其他系統(tǒng)下也一樣氮双。一、getchar的兩點總結:1.getchar是以行為單位進行存取的霎匈。當用getchar進行輸入時戴差,如果輸入的第一個字符為有效字符(即輸入是文件結束符EOF,Windows下為組合鍵Ctrl+Z唧躲,Unix/Linux下為組合鍵Ctrl+D)造挽,那么只有當最后一個輸入字符為換行符'\n'(也可以是文件結束符EOF碱璃,EOF將在后面討論)時,getchar才會停止執(zhí)行饭入,整個程序將會往下執(zhí)行嵌器。譬如下面程序段:while
(
(
c =
getchar
(
)
)
!
=
EOF
)
{
putchar
(
c)
;
}

執(zhí)行程序,輸入:abc谐丢,然后回車爽航。則程序就會去執(zhí)行puchar(c),然后輸出abc乾忱,這個地方不要忘了讥珍,系統(tǒng)輸出的還有一個回車。然后可以繼續(xù)輸入窄瘟,再次遇到換行符的時候衷佃,程序又會把那一行的輸入的字符輸出在終端上。對于getchar蹄葱,肯定很多初學的朋友會問氏义,getchar不是以字符為單位讀取的嗎?那么图云,既然我輸入了第一個字符a惯悠,肯定滿足while循環(huán)(c = getchar()) != EOF的條件阿,那么應該執(zhí)行putchar(c)在終端輸出一個字符a竣况。不錯克婶,我在用getchar的時候也是一直這么想的,但是程序就偏偏不著樣執(zhí)行丹泉,而是必需讀到一個換行符或者文件結束符EOF才進行一次輸出情萤。對這個問題的一個解釋是,在大師編寫C的時候嘀掸,當時并沒有所謂終端輸入的概念紫岩,所有的輸入實際上都是按照文件進行讀取的,文件中一般都是以行為單位的睬塌。因此,只有遇到換行符歇万,那么程序會認為輸入結束揩晴,然后采取執(zhí)行程序的其他部分。同時贪磺,輸入是按照文件的方式存取的硫兰,那么要結束一個文件的輸入就需用到EOF(Enf Of File). 這也就是為什么getchar結束輸入退出時要用EOF的原因。2.getchar()的返回值一般情況下是字符寒锚,但也可能是負值劫映,即返回EOF违孝。這里要強調的一點就是,getchar函數通常返回終端所輸入的字符泳赋,這些字符系統(tǒng)中對應的ASCII值都是非負的雌桑。因此,很多時候祖今,我們會寫這樣的兩行代碼:char
c;
c =
getchar
(
)
;

這樣就很有可能出現問題校坑。因為getchar函數除了返回終端輸入的字符外,在遇到Ctrl+D(Linux下)即文件結束符EOF時千诬,getchar()的返回EOF耍目,這個EOF在函數庫里一般定義為-1。因此徐绑,在這種情況下邪驮,getchar函數返回一個負值,把一個負值賦給一個char型的變量是不正確的傲茄。為了能夠讓所定義的變量能夠包含getchar函數返回的所有可能的值毅访,正確的定義方法如下(K&R C中特別提到了這個問題):int
c;
c =
getchar
(
)
;

二、EOF的兩點總結(主要指普通終端中的EOF)1.EOF作為文件結束符時的情況:EOF雖然是文件結束符烫幕,但并不是在任何情況下輸入Ctrl+D(Windows下Ctrl+Z)都能夠實現文件結束的功能俺抽,只有在下列的條件下,才作為文件結束符较曼。(1)遇到getcahr函數執(zhí)行時磷斧,要輸入第一個字符時就直接輸入Ctrl+D,就可以跳出getchar(),去執(zhí)行程序的其他部分捷犹;(2)在前面輸入的字符為換行符時弛饭,接著輸入Ctrl+D;(3)在前面有字符輸入且不為換行符時萍歉,要連著輸入兩次Ctrl+D侣颂,這時第二次輸入的Ctrl+D起到文件結束符的功能,至于第一次的Ctrl+D的作用將在下面介紹枪孩。其實憔晒,這三種情況都可以總結為只有在getchar()提示新的一次輸入時,直接輸入Ctrl+D才相當于文件結束符蔑舞。2.EOF作為行結束符時的情況拒担,這時候輸入Ctrl+D并不能結束getchar(),而只能引發(fā)getchar()提示下一輪的輸入。這種情況主要是在進行getchar()新的一行輸入時攻询,當輸入了若干字符(不能包含換行符)之后从撼,直接輸入Ctrl+D,此時的Ctrl+D并不是文件結束符钧栖,而只是相當于換行符的功能低零,即結束當前的輸入婆翔。以上面的代碼段為例,如果執(zhí)行時輸入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()的執(zhí)行刁愿。【補充】本文第二部分中關于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
(
)
; /*

終端驅動處又回到一次一行模式

*/
system("stty cooked"); return 0;}

編譯運行該程序停团,則當如入一個字符時旷坦,直接出處一個字符掏熬,然后程序結束。由此可見秒梅,由于終端驅動的模式不同旗芬,造成了getchar()輸入結束的條件不一樣。普通模式下需要回車或者EOF捆蜀,而在一次一個字符的模式下疮丛,則輸入一個字符之后就結束了。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末辆它,一起剝皮案震驚了整個濱河市誊薄,隨后出現的幾起案子,更是在濱河造成了極大的恐慌锰茉,老刑警劉巖呢蔫,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現場離奇詭異飒筑,居然都是意外死亡片吊,警方通過查閱死者的電腦和手機,發(fā)現死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進店門协屡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來俏脊,“玉大人,你說我怎么就攤上這事肤晓∫叮” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵材原,是天一觀的道長沸久。 經常有香客問我,道長余蟹,這世上最難降的妖魔是什么卷胯? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮威酒,結果婚禮上窑睁,老公的妹妹穿的比我還像新娘。我一直安慰自己葵孤,他們只是感情好担钮,可當我...
    茶點故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著尤仍,像睡著了一般箫津。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天苏遥,我揣著相機與錄音饼拍,去河邊找鬼。 笑死田炭,一個胖子當著我的面吹牛师抄,可吹牛的內容都是我干的。 我是一名探鬼主播教硫,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼叨吮,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了瞬矩?” 一聲冷哼從身側響起茶鉴,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎丧鸯,沒想到半個月后蛤铜,有當地人在樹林里發(fā)現了一具尸體,經...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡丛肢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年围肥,在試婚紗的時候發(fā)現自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蜂怎。...
    茶點故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡穆刻,死狀恐怖,靈堂內的尸體忽然破棺而出杠步,到底是詐尸還是另有隱情氢伟,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布幽歼,位于F島的核電站朵锣,受9級特大地震影響,放射性物質發(fā)生泄漏甸私。R本人自食惡果不足惜诚些,卻給世界環(huán)境...
    茶點故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望皇型。 院中可真熱鬧诬烹,春花似錦、人聲如沸弃鸦。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽唬格。三九已至家破,卻和暖如春颜说,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背员舵。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工脑沿, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人马僻。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像注服,于是被迫代替她去往敵國和親韭邓。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 42,762評論 2 345

推薦閱讀更多精彩內容