本文摘自:http://www.jb51.net/article/36848.htm
一菇篡、getchar的兩點(diǎn)總結(jié):
1.getchar是以行為單位進(jìn)行存取的拯杠。
當(dāng)用getchar進(jìn)行輸入時(shí)速和,如果輸入的第一個(gè)字符為有效字符(即輸入是文件結(jié)束符EOF圆雁,Windows下為組合鍵Ctrl+Z地淀, Unix/Linux下為組合鍵Ctrl+D),那么只有當(dāng)最后一個(gè)輸入字符為換行符'\n'(也可以是文件結(jié)束符EOF臼寄,EOF將在后面討論)時(shí)霸奕, getchar才會(huì)停止執(zhí)行,整個(gè)程序?qū)?huì)往下執(zhí)行吉拳。譬如下面程序段:
while((c = getchar()) != EOF){
putchar(c);
}
執(zhí)行程序质帅,輸入:abc,然后回車(chē)留攒。則程序就會(huì)去執(zhí)行puchar(c)煤惩,然后輸出abc,這個(gè)地方不要忘了炼邀,系統(tǒng)輸出的還有一個(gè)回車(chē)魄揉。然后可以繼續(xù)輸入,再次遇到換行符的時(shí)候拭宁,程序又會(huì)把那一行的輸入的字符輸出在終端上洛退。
對(duì)于getchar,肯定很多初學(xué)的朋友會(huì)問(wèn)红淡,getchar不是以字符為單位讀取的嗎不狮?那么,既然我輸入了第一個(gè)字符a在旱,肯定滿足while循環(huán)(c = getchar()) != EOF的條件阿摇零,那么應(yīng)該執(zhí)行putchar(c)在終端輸出一個(gè)字符a。不錯(cuò)桶蝎,我在用getchar的時(shí)候也是一直這么想的驻仅,但是程序就偏偏不著樣執(zhí) 行,而是必需讀到一個(gè)換行符或者文件結(jié)束符EOF才進(jìn)行一次輸出登渣。
對(duì)這個(gè)問(wèn)題的一個(gè)解釋是噪服,在大師編寫(xiě)C的時(shí)候,當(dāng)時(shí)并沒(méi)有所謂終端輸入的概念胜茧,所有的輸入實(shí)際上都是按照文件進(jìn)行讀取的粘优,文件中一般都是以行為單位的。因 此呻顽,只有遇到換行符雹顺,那么程序會(huì)認(rèn)為輸入結(jié)束,然后采取執(zhí)行程序的其他部分廊遍。同時(shí)嬉愧,輸入是按照文件的方式存取的,那么要結(jié)束一個(gè)文件的輸入就需用到EOF (Enf Of File). 這也就是為什么getchar結(jié)束輸入退出時(shí)要用EOF的原因喉前。
2.getchar()的返回值一般情況下是字符没酣,但也可能是負(fù)值王财,即返回EOF。
這里要強(qiáng)調(diào)的一點(diǎn)就是裕便,getchar函數(shù)通常返回終端所輸入的字符绒净,這些字符系統(tǒng)中對(duì)應(yīng)的ASCII值都是非負(fù)的。因此偿衰,很多時(shí)候疯溺,我們會(huì)寫(xiě)這樣的兩行代碼:
char c;
c = getchar();
這樣就很有可能出現(xiàn)問(wèn)題。因?yàn)間etchar函數(shù)除了返回終端輸入的字符外哎垦,在遇到Ctrl+D(Linux下)即文件結(jié)束符EOF時(shí)囱嫩,getchar ()的返回EOF,這個(gè)EOF在函數(shù)庫(kù)里一般定義為-1漏设。因此墨闲,在這種情況下,getchar函數(shù)返回一個(gè)負(fù)值郑口,把一個(gè)負(fù)值賦給一個(gè)char型的變量是不 正確的鸳碧。為了能夠讓所定義的變量能夠包含getchar函數(shù)返回的所有可能的值,正確的定義方法如下(K&R C中特別提到了這個(gè)問(wèn)題):
int c;
c = getchar();
二犬性、EOF的兩點(diǎn)總結(jié)(主要指普通終端中的EOF)
1.EOF作為文件結(jié)束符時(shí)的情況:
EOF雖然是文件結(jié)束符瞻离,但并不是在任何情況下輸入Ctrl+D(Windows下Ctrl+Z)都能夠?qū)崿F(xiàn)文件結(jié)束的功能,只有在下列的條件下乒裆,才作為文件結(jié)束符套利。
(1)遇到getcahr函數(shù)執(zhí)行時(shí),要輸入第一個(gè)字符時(shí)就直接輸入Ctrl+D鹤耍,就可以跳出getchar(),去執(zhí)行程序的其他部分肉迫;
(2)在前面輸入的字符為換行符時(shí),接著輸入Ctrl+D稿黄;
(3)在前面有字符輸入且不為換行符時(shí)喊衫,要連著輸入兩次Ctrl+D,這時(shí)第二次輸入的Ctrl+D起到文件結(jié)束符的功能杆怕,至于第一次的Ctrl+D的作用將在下面介紹族购。
其實(shí),這三種情況都可以總結(jié)為只有在getchar()提示新的一次輸入時(shí)陵珍,直接輸入Ctrl+D才相當(dāng)于文件結(jié)束符寝杖。
2.EOF作為行結(jié)束符時(shí)的情況,這時(shí)候輸入Ctrl+D并不能結(jié)束getchar(),而只能引發(fā)getchar()提示下一輪的輸入撑教。
這種情況主要是在進(jìn)行g(shù)etchar()新的一行輸入時(shí)朝墩,當(dāng)輸入了若干字符(不能包含換行符)之后醉拓,直接輸入Ctrl+D伟姐,此時(shí)的Ctrl+D并不是文件 結(jié)束符收苏,而只是相當(dāng)于換行符的功能,即結(jié)束當(dāng)前的輸入愤兵。以上面的代碼段為例鹿霸,如果執(zhí)行時(shí)輸入abc,然后Ctrl+D秆乳,程序輸出結(jié)果為:
abcabc
注意:第一組abc為從終端輸入的懦鼠,然后輸入Ctrl+D,就輸出第二組abc屹堰,同時(shí)光標(biāo)停在第二組字符的c后面,然后可以進(jìn)行新一次的輸入肛冶。這時(shí)如果再次輸入Ctrl+D,則起到了文件結(jié)束符的作用扯键,結(jié)束getchar()睦袖。
如果輸入abc之后,然后回車(chē)荣刑,輸入換行符的話馅笙,則終端顯示為:
abc //第一行,帶回車(chē)
abc //第二行
//第三行
其中第一行為終端輸入厉亏,第二行為終端輸出董习,光標(biāo)停在了第三行處,等待新一次的終端輸入爱只。
從這里也可以看出Ctrl+D和換行符分別作為行結(jié)束符時(shí)皿淋,輸出的不同結(jié)果。
EOF的作用也可以總結(jié)為:當(dāng)終端有字符輸入時(shí)恬试,Ctrl+D產(chǎn)生的EOF相當(dāng)于結(jié)束本行的輸入沥匈,將引起getchar()新一輪的輸入;當(dāng)終端沒(méi)有字符 輸入或者可以說(shuō)當(dāng)getchar()讀取新的一次輸入時(shí)忘渔,輸入Ctrl+D高帖,此時(shí)產(chǎn)生的EOF相當(dāng)于文件結(jié)束符,程序?qū)⒔Y(jié)束getchar()的執(zhí)行畦粮。