大師級經典的著做,要字斟句酌的去讀,去理解。之前在看K&R的The C Programming Language(SecondEdition) 第1.5節的字符輸入/輸出,被getchar()和EOF所迷惑了。可能主要還是因爲沒有搞清楚getchar()的工做原理和EOF的用法。所以,感覺頗有必要總結一下,否則,不少瑣碎的知識點長時間過後就會淡忘的,只有寫下來纔是最好的方法。 其實,getchar()最典型的程序也就幾行代碼而已。本人所用的環境是DebianGNU/Linux,在其餘系統下也一樣。 1、getchar的兩點總結: 1.getchar是以行爲單位進行存取的。 當用getchar進行輸入時,若是輸入的第一個字符爲有效字符(即輸入是文件結束符EOF,Windows下爲組合鍵Ctrl+Z, Unix/Linux下爲組合鍵Ctrl+D),那麼只有當最後一個輸入字符爲換行符'\n'(也能夠是文件結束符EOF,EOF將在後面討論)時, getchar纔會中止執行,整個程序將會往下執行。譬以下面程序段: spa
while((c = getchar()) != EOF){ putchar(c); } orm |
執行程序,輸入:abc,而後回車。則程序就會去執行puchar(c),而後輸出abc,這個地方不要忘了,系統輸出的還有一個回車。而後能夠繼續輸入,再次遇到換行符的時候,程序又會把那一行的輸入的字符輸出在終端上。 對於getchar,確定不少初學的朋友會問,getchar不是以字符爲單位讀取的嗎?那麼,既然我輸入了第一個字符a,確定滿足while循環(c = getchar()) != EOF的條件阿,那麼應該執行putchar(c)在終端輸出一個字符a。不錯,我在用getchar的時候也是一直這麼想的,可是程序就恰恰不着樣執行,而是必需讀到一個換行符或者文件結束符EOF才進行一次輸出。 對這個問題的一個解釋是,在大師編寫C的時候,當時並無所謂終端輸入的概念,全部的輸入實際上都是按照文件進行讀取的,文件中通常都是以行爲單位的。所以,只有遇到換行符,那麼程序會認爲輸入結束,而後採起執行程序的其餘部分。同時,輸入是按照文件的方式存取的,那麼要結束一個文件的輸入就需用到EOF (Enf Of File). 這也就是爲什麼getchar結束輸入退出時要用EOF的緣由。 2.getchar()的返回值通常狀況下是字符,但也多是負值,即返回EOF。 這裏要強調的一點就是,getchar函數一般返回終端所輸入的字符,這些字符系統中對應的ASCII值都是非負的。所以,不少時候,我們會寫這樣的兩行代碼: ci
char c; c = getchar(); get |
這樣就頗有可能出現問題。因爲getchar函數除了返回終端輸入的字符外,在遇到Ctrl+D(Linux下)即文件結束符EOF時,getchar ()的返回EOF,這個EOF在函數庫裏通常定義爲-1。所以,在這種狀況下,getchar函數返回一個負值,把一個負值賦給一個char型的變量是不正確的。爲了能夠讓所定義的變量能夠包含getchar函數返回的全部可能的值,正確的定義方法以下(K&R C中特別提到了這個問題):it
2、EOF的兩點總結(主要指普通終端中的EOF) 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和換行符分別做爲行table |