c++ 輸入函數(shù)的比較和運(yùn)用
cin, cin.get(),cin.getline()
文章分類:C++編程cin讀入數(shù)據(jù)遇到空格結(jié)束;并且丟棄空格符,輸入遇到回車符結(jié)束,很正常。
二.cin.get()
該函數(shù)有三種格式:無(wú)參,一參數(shù),二參數(shù)
即cin.get(), cin.get(char ch), cin.get(array_name, Arsize)
讀取字符的情況:
輸入結(jié)束條件:Enter鍵
對(duì)結(jié)束符處理:不丟棄緩沖區(qū)中的Enter
cin.get() 與 cin.get(char ch)用于讀取字符,他們的使用是相似的,
即:ch=cin.get() 與 cin.get(ch)是等價(jià)的
<
C/C++學(xué)習(xí)筆記1 - 深入了解scanf()/getchar()和gets()等函數(shù)
http://blog.csdn.net/lewsn2008
----------------------------------------------------
| 問題描述一:(分析scanf()和getchar()讀取字符) |
----------------------------------------------------
scanf(), getchar()等都是標(biāo)準(zhǔn)輸入函數(shù),一般人都會(huì)覺得這幾個(gè)函數(shù)非常簡(jiǎn)單,沒什么特殊的。但是有時(shí)候卻就是因?yàn)槭褂眠@些函數(shù)除了問題,卻找不出其中的原因。下面先看一個(gè)很簡(jiǎn)單的程序:
程序1:
#include
int main()
{
char ch1, ch2;
scanf("%c", &ch1);
scanf("%c", &ch2);
printf("%d %d ", ch1, ch2);
return 0;
}
或者是:
#include
int main()
{
char ch1, ch2;
ch1 = getchar();
ch2 = getchar();
printf("%d %d ", ch1, ch2);
return 0;
}
程序的本意很簡(jiǎn)單,就是從鍵盤讀入兩個(gè)字符,然后打印出這兩個(gè)字符的ASCII碼值。可是執(zhí)行程序后會(huì)發(fā)現(xiàn)除了問題:當(dāng)從鍵盤輸入一個(gè)字符后,就打印出了結(jié)果,根本就沒有輸入第二個(gè)字符程序就結(jié)束了。例如用戶輸入字符'a', 打印結(jié)果是97,10。這是為什么呢?
【分析】:
首先我們呢看一下輸入操作的原理,程序的輸入都建有一個(gè)緩沖區(qū),即輸入緩沖區(qū)。一次輸入過程是這樣的,當(dāng)一次鍵盤輸入結(jié)束時(shí)會(huì)將輸入的數(shù)據(jù)存入輸入緩沖區(qū),而cin函數(shù)直接從輸入緩沖區(qū)中取數(shù)據(jù)。正因?yàn)閏in函數(shù)是直接從緩沖區(qū)取數(shù)據(jù)的,所以有時(shí)候當(dāng)緩沖區(qū)中有殘留數(shù)據(jù)時(shí),cin函數(shù)會(huì)直接取得這些殘留數(shù)據(jù)而不會(huì)請(qǐng)求鍵盤輸入,這就是例子中為什么會(huì)出現(xiàn)輸入語(yǔ)句失效的原因!
其實(shí)這里的10恰好是回車符!這是因?yàn)閟canf()和getchar()函數(shù)是從輸入流緩沖區(qū)中讀取值的,而并非從鍵盤(也就是終端)緩沖區(qū)讀取。而讀取時(shí)遇到回車( )而結(jié)束的,這個(gè)會(huì)一起讀入輸入流緩沖區(qū)的,所以第一次接受輸入時(shí)取走字符后會(huì)留下字符,這樣第二次的讀入函數(shù)直接從緩沖區(qū)中把取走了,顯然讀取成功了,所以不會(huì)再?gòu)慕K端讀取!這就是為什么這個(gè)程序只執(zhí)行了一次輸入操作就結(jié)束的原因!
----------------------------------------------------
| 問題描述二:(分析scanf()和gets()讀取字符串) |
----------------------------------------------------
首先我們看一下scanf()讀取字符串的問題:
程序2:
#include
int main()
{
char str1[20], str2[20];
scanf("%s",str1);
printf("%s ",str1);
scanf("%s",str2);
printf("%s ",str2);
return 0;
}
程序的功能是讀入一個(gè)字符串輸出,在讀入一個(gè)字符串輸出。可我們會(huì)發(fā)現(xiàn)輸入的字符串中不能出現(xiàn)空格,例如:
測(cè)試一輸入:
Hello world!
輸出:
Hello
world!
【分析】到此程序執(zhí)行完畢,不會(huì)執(zhí)行第二次的讀取操作!這個(gè)問題的原因跟問題一類似,第一次輸入Hello world!后,字符串Hello world!都會(huì)被讀到輸入緩沖區(qū)中,而scanf()函數(shù)取數(shù)據(jù)是遇到回車、空格、TAB就會(huì)停止,也就是第一個(gè)scanf()會(huì)取出"Hello",而"world!"還在緩沖區(qū)中,這樣第二個(gè)scanf會(huì)直接取出這些數(shù)據(jù),而不會(huì)等待從終端輸入。
測(cè)試二:
Hello[Enter]
Hello[輸出]
world[Enter]
world[輸出]
【分析】程序執(zhí)行了兩次從鍵盤讀入字符串,說(shuō)明第一次輸入結(jié)束時(shí)的回車符被丟棄!即:scanf()讀取字符串會(huì)舍棄最后的回車符!
我們?cè)倏匆幌耮ets()讀取字符串的情況:
用scanf來(lái)讀取一個(gè)字符串時(shí),字符串中是不可以出現(xiàn)空格的,一旦出現(xiàn)空格,后面的數(shù)據(jù)就會(huì)舍棄殘留在緩沖區(qū)中。其實(shí)有另外一個(gè)函數(shù)是可以接受空格的,那就是gets(),下面我們看一下這個(gè)函數(shù)的應(yīng)用,我們把程序2改動(dòng)一下:
程序3:
#include
int main()
{
char str1[20], str2[20];
gets(str1);
printf("%s ",str1);
gets(str2);
printf("%s ",str2);
return 0;
}
測(cè)試:
Hello world! [輸入]
Hello world! [輸出]
12345 [輸入]
12345 [輸出]
【分析】顯然與上一個(gè)程序的執(zhí)行情況不同,這次程序執(zhí)行了兩次從鍵盤的讀入,而且第一個(gè)字符串取了Hello world! 接受了空格符,而沒有像上一個(gè)程序那樣分成了兩個(gè)字符串!所以如果要讀入一個(gè)帶空格符的字符串時(shí)因該用gets(), 而不宜用scanf()!
--------------------------------------------------------
| 問題描述三:(getchar()暫停程序,查看程序執(zhí)行結(jié)果)|
--------------------------------------------------------
不知道大家有沒有遇到過這樣的問題,有的編譯器程序執(zhí)行完后的結(jié)果界面不會(huì)停下而是一閃就沒了,以至于看不到執(zhí)行結(jié)果。所以很多人在程序最后加上 getchar()語(yǔ)句,目的是想讓程序執(zhí)行完后停下來(lái),等待從終端接收一個(gè)字符再結(jié)束程序。可是發(fā)現(xiàn)有時(shí)候這樣根本沒用,程序照樣跳出去了。這是為什么呢?
【分析】原因跟上面例子講的一樣,是因?yàn)檩斎刖彌_區(qū)中還有數(shù)據(jù),所以getchar()會(huì)成果讀到數(shù)據(jù),所以就跳出了!
------------------
| 【總結(jié)】 |
------------------
第一: 要注意不同的函數(shù)是否接受空格符、是否舍棄最后的回車符的問題!
讀取字符時(shí):
scanf()以Space、Enter、Tab結(jié)束一次輸入,不會(huì)舍棄最后的回車符(即回車符會(huì)殘留在緩沖區(qū)中);
getchar()以Enter結(jié)束輸入,也不會(huì)舍棄最后的回車符;
讀取字符串時(shí):
scanf()以Space、Enter、Tab結(jié)束一次輸入
gets()以Enter結(jié)束輸入(空格不結(jié)束),接受空格,會(huì)舍棄最后的回車符!
第二: 為了避免出現(xiàn)上述問題,必須要清空緩沖區(qū)的殘留數(shù)據(jù),可以用以下的方法解決:
方法1:C語(yǔ)言里提供了函數(shù)清空緩沖區(qū),只要在讀數(shù)據(jù)之前先清空緩沖區(qū)就沒問題了!
這個(gè)函數(shù)是fflush(stdin)。
方法2:自己取出緩沖區(qū)里的殘留數(shù)據(jù)。
(說(shuō)實(shí)話這個(gè)語(yǔ)句我也沒看懂,呵呵!為什么格式控制是這樣的!希望高手指點(diǎn)一下!)
scanf("%[^ ]",string);
posted on 2011-05-28 22:08 pp_zhang 閱讀(1447) 評(píng)論(0) 編輯 收藏 引用 所屬分類: c/c++