[轉載]C語言 getchar()與EOF

本文轉自http://cunlau.cnblogs.com/,謝謝原做者。

大師級經典的著做,要字斟句酌的去讀,去理解。之前在看K&R的The C Programming Language(Second Edition)中第1.5節的字符輸入/輸出,很迷惑getchar()和EOF的行爲。所以,感受頗有必要總結一下,否則,不少瑣碎的知識點長時間事後就會淡忘的,只有寫下來纔是最好的方法。

1、對getchar的兩點總結:函數

1. getchar是以行爲單位進行存取的。
當調用getchar函數讀取輸入時,只有當輸入字符爲換行符'/n'或文件結束符EOF時,getchar纔會中止執行,整個程序將會往下執行。而且,若是輸入行是以EOF結束的(EOF以前不是換行符),則EOF會被「吃掉」(即不會被getchar讀取到)。譬以下面程序段:post

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

執行程序,輸入:abc,而後回車。則程序就會去執行puchar(c),而後輸出abc和一個回車。而後能夠繼續輸入,再次遇到換行符的時候,程序又會把 那一行的輸入的字符輸出在終端上。使人迷惑的是,getchar不是以字符爲單位讀取的嗎?那麼,既然我輸入了第一個字符a,確定知足while循環(c = getchar()) != EOF的條件,那麼應該執行putchar(c)在終端輸出一個字符a。可是程序就恰恰不這樣執行,而是必需讀到一個換行符或者文件結束符EOF才進行一次輸出。
 
形成這種結果的一種解釋是,輸入終端驅動處於一次一行 的模式下。也就是雖然getchar()和putchar()確實是按照每次一個字符進行的。可是終端驅動處於一次一行的模式,它的輸入只有到'/n'或 者EOF時才結束。在本例中,程序段調用了getchar函數,則控制權從程序段轉移到getchar函數,而getchar函數要依賴於操做系統的驅動 來讀取輸入,沒遇到換行符或者EOF,驅動不會通知getchar函數,getchar函數處於「阻 塞」狀態。而遇到換行符或者EOF後,getchar函數解除「阻塞」,讀取一個字符,控制權返回程 序段,執行putchar函數,循環執行。直到遇到EOF字符或者這行輸入所有處理完。操作系統


2. getchar()的返回值通常狀況下是非負 值,但也多是負值,即返回EOF。這個EOF在函數庫裏通常定義爲-1。正確的定義方法以下(K&R C中特別提到了這個問題):blog

int c;
c = getchar();get

2、EOF的兩點總結(主要指普通終端中的EOF)
1. EOF做爲文件結束符時的狀況:
EOF雖然是文件結束符,但並非在任何狀況下輸入Ctrl+D(Windows下Ctrl+Z)都可以實現文件結束的功能,只有在下列的條件下,才做爲文件結束符。
(1)遇到getcahr函數執行時,要輸入第一個字符時就直接輸入Ctrl+D;
(2)在前面輸入的字符爲換行符時, 接着輸入Ctrl+D;
(3)在前面有字符輸入且不爲換行符時,要連着輸入兩次Ctrl+D,這時第二次輸入的Ctrl+D起到文件結束符的功能,至於第一次的
Ctrl+D做爲行結束符(如1.1所講)。博客

其實,這三種狀況均可以總結爲只有在getchar()提示新的一次輸入時, 直接輸入Ctrl+D才至關於文件結束符。it

2. EOF做爲行結束符時的狀況,這時候輸入Ctrl+D做爲行結束的標誌能結束getchar()的「阻塞」,使getchar()逐個字符讀入,可是EOF會被「吃掉」,並不會被讀取。
 
以上面的代碼段爲例, 若是執行時輸入abc,而後 Ctrl+D,程序輸出結果爲:
abcabc
注意:第一組abc是你從終端輸入的,而後輸入Ctrl+D,getchar逐個字符讀取並逐個輸出打印出第二組abc,同時光標停在第二組字符的c後面,而後能夠進行新一次的輸入。這時若是再次輸入Ctrl+D,就會起到了文件結束符的做用,由於EOF是一行輸入的第一個字符。若是輸入abc以後,而後回車,輸入換行符的話,則終端顯示爲:
abc'/n'
abc'/n'
//第三行io

其中第一行爲你是終端輸入的,第二行是終端輸出(含換行符),光標停在了第三行處,等待新一次的終端輸入。從這裏也 能夠看出Ctrl+D和換行符分別做爲行結束符時,輸出的不一樣結果。class

 

若是認爲此文對您有幫助,別忘了支持一下哦!

做者:竹影清風
聲明:本文版權歸做者和博客園共有,歡迎轉載,但未經做者贊成必須保留此段聲明,且在文章頁面明顯位置給出原文鏈接,不然保留追究法律責任的權利。若有問題或建議,請多多賜教,很是感謝。
相關文章
相關標籤/搜索