C++字符串輸入緩沖區(qū)機(jī)制詳解
一、緩沖定義
1.緩沖定義
緩沖是在兩種不同速度設(shè)備之間傳輸信息時(shí)平滑傳輸過(guò)程的常用手段。
2.為什么引入緩沖區(qū)
操作系統(tǒng)這門(mén)課有明確的說(shuō)明緩沖的作用,是為了解決高速設(shè)備和低速設(shè)備之間速度不匹配的問(wèn)題,直接舉個(gè)書(shū)上的CPU和打印機(jī)的例子:
首先我們要用打印機(jī)打印一篇文章肯定要經(jīng)過(guò)CPU處理然后給到打印機(jī)打印文章,但是CPU處理1000個(gè)字節(jié)的文字一眨眼的事情,但是打印機(jī)可能需要若干秒,總不可能讓CPU處理一個(gè)數(shù)據(jù)給打印機(jī)一個(gè)數(shù)據(jù)這樣子等著運(yùn)行,這就引入了緩沖區(qū),CPU處理完數(shù)據(jù)存入緩沖區(qū),打印機(jī)直接從緩沖區(qū)提取已經(jīng)處理好的數(shù)據(jù),這樣子就解決了高速設(shè)備(CPU)和低速設(shè)備(打印機(jī))處理速度不匹配的問(wèn)題。
二、scanf,cin輸入緩沖區(qū)
1.scanf和cin的緩沖類型
scanf和cin的緩沖類型為行緩沖,行緩沖的的特點(diǎn)是在輸入數(shù)據(jù)只要沒(méi)有碰到換行符(回車)就將數(shù)據(jù)存入輸入緩沖區(qū),當(dāng)碰到換行符之后就將緩沖區(qū)中的數(shù)據(jù)取出使用。
2.scanf和cin的緩沖機(jī)制
scanf和cin輸入數(shù)據(jù)緩沖機(jī)制基本一致,在讀入一個(gè)數(shù)據(jù)時(shí)直到回車之前他都會(huì)存儲(chǔ)在輸入緩沖區(qū)中,直到碰到回車才會(huì)將數(shù)據(jù)從輸入緩沖區(qū)中取出供變量使用,但是緩沖區(qū)中的換行符會(huì)被留在輸入緩沖區(qū)中。
3.cin.getline和cin.get
cin.get讀取字符串直到讀取到回車為止,但是也會(huì)將回車留在緩沖區(qū)。
cin.getline讀取字符串直到讀取到回車為止,但是不會(huì)講回車留在緩沖區(qū)。
4.scanf和cin輸入
①cin和scanf讀取一個(gè)字符的區(qū)別
char c;
cin>>c;//cin讀取字符的時(shí)候不會(huì)讀入空格、回車以及制表符,如果緩沖區(qū)開(kāi)頭是換行符或者制表符會(huì)被忽略并清除
scanf("%c",&c);//scanf讀取字符的時(shí)候不會(huì)管你是什么字符直接讀進(jìn)來(lái)
②cin和scanf在讀取緩沖區(qū)中的數(shù)字、字符串、浮點(diǎn)型的時(shí)候不會(huì)將開(kāi)頭的空格、換行符或者制表符當(dāng)作數(shù)據(jù)讀入,如果碰到了他們會(huì)將它們忽略并清除。需要注意cin.get和cin.getline不會(huì)將開(kāi)頭的空格、換行符或者制表符忽略并刪除。
5.可能遇到的問(wèn)題
①在用cin或者scanf讀取一個(gè)字符串之后scanf再讀取一個(gè)字符,發(fā)現(xiàn)讀取的字符沒(méi)辦法輸入而且輸出了一個(gè)回車??慈缦麓a:
char name[20];
char c;
cin>>name;
cout<<name<<endl;
scanf("%c",&c);
cout<<"---"<<int(c)<<endl;
結(jié)果如下截圖:

當(dāng)我們輸入name為123的時(shí)候程序就結(jié)束了輸出c的值為10,在ASCII碼編碼中10對(duì)應(yīng)的是換行符,為什么會(huì)出現(xiàn)上述這種情況呢?因?yàn)閏in輸入完之后將回車保留在了緩沖區(qū)中,而scanf("%c",&c)輸入字符并不會(huì)判斷是空格還是回車,直接將緩沖區(qū)中的回車拿了出來(lái)給了字符變量c。
②用cin.getline輸入字符串之后再用scanf讀取一個(gè)字符就和上面不一樣了??慈缦麓a:
char name[20];
char c;
cin.getline(name,20);
cout<<name<<endl;
scanf("%c",&c);
cout<<"---"<<int(c)<<endl;
結(jié)果如下截圖:

結(jié)果不同的原因是因?yàn)閟canf,cin,cin.get在行緩沖取出數(shù)據(jù)之后會(huì)將換行符留在緩沖區(qū)中,然后再用scanf讀入一個(gè)字符發(fā)現(xiàn)緩沖區(qū)并不為空就從緩沖區(qū)中將數(shù)據(jù)拿出來(lái),而cin.getline會(huì)將緩沖區(qū)中的換行符也清除,所以不會(huì)有①這種情況出現(xiàn)。
③讀入一個(gè)字符串之后后面的cin.get()一直無(wú)法讀取數(shù)據(jù),看如下代碼:
char name[20];
cin.get(name,20);
cout<<"--"<<name<<endl;
cin.get(name,20);
cout<<"--"<<name<<endl;
cin.get(name,20);
cout<<"--"<<name<<endl;
cin.get(name,20);
cout<<"--"<<name<<endl;
cin.get(name,20);
cout<<"--"<<name<<endl;
結(jié)果如下截圖:

可以看到用第一個(gè)cin.get輸入了數(shù)據(jù)之后后面的cin.get都無(wú)效了,這是因?yàn)閏in.get從緩沖區(qū)中讀取到換行符就結(jié)束并將換行符保留在緩沖區(qū)中,接下來(lái)的cin.get一直在緩沖區(qū)中碰到換行符就一直沒(méi)有實(shí)際數(shù)據(jù)輸入。
④整形和字符串混合輸入
當(dāng)先輸入一個(gè)整形再用cin.getline輸入字符串會(huì)發(fā)現(xiàn)沒(méi)有經(jīng)歷輸入字符串的過(guò)程,測(cè)試如下代碼:
char name[20];
int a;
cin>>a;
cout<<a<<endl;
cin.getline(name,20);
cout<<"---"<<name<<endl;
cout<<"end"<<endl;
結(jié)果如下:

根據(jù)上面的緩沖原理比較容易理解,cin輸入一個(gè)整形后會(huì)在緩沖區(qū)中留下一個(gè)換行符,因?yàn)榫彌_區(qū)中有換行符剩余,cin.getline從緩沖區(qū)中取出數(shù)據(jù)發(fā)現(xiàn)正好是換行符,那么就默認(rèn)字符串輸入完了,并將換行符從緩沖區(qū)中去掉,所以造成了上面的局面。
清除緩沖區(qū)
上面講的問(wèn)題基本上都是因?yàn)榫彌_區(qū)中剩余的數(shù)據(jù)造成的,那么只需要有方法清除緩沖區(qū)就行了,可以用getchar(),cin.get()讀取緩沖區(qū)中多的換行符。
總結(jié)
本篇文章就到這里了,希望能夠給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
C語(yǔ)言學(xué)生學(xué)籍管理系統(tǒng)課程設(shè)計(jì)
這篇文章主要介紹了C語(yǔ)言學(xué)生學(xué)籍管理系統(tǒng)課程設(shè)計(jì),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-01-01
C++設(shè)計(jì)模式之工廠方法模式的實(shí)現(xiàn)及優(yōu)缺點(diǎn)
工廠方法模式是一個(gè)創(chuàng)建型設(shè)計(jì)模式,通過(guò)定義一個(gè)創(chuàng)建對(duì)象的接口,讓其子類決定實(shí)例化哪一個(gè)工廠類,這篇文章主要給大家介紹了關(guān)于C++設(shè)計(jì)模式之工廠方法模式的實(shí)現(xiàn)及優(yōu)缺點(diǎn),需要的朋友可以參考下2021-06-06
VC實(shí)現(xiàn)的病毒專殺工具完整實(shí)例
這篇文章主要介紹了VC實(shí)現(xiàn)的病毒專殺工具完整實(shí)例,詳細(xì)講述了針對(duì)病毒的進(jìn)程終止、刪除文件及回復(fù)注冊(cè)表與啟動(dòng)項(xiàng)等,同時(shí)介紹了與之相關(guān)的系統(tǒng)函數(shù),非常具有參考借鑒價(jià)值,需要的朋友可以參考下2014-10-10
C語(yǔ)言中使用qsort函數(shù)對(duì)自定義結(jié)構(gòu)體數(shù)組進(jìn)行排序
這篇文章主要介紹了C語(yǔ)言中使用qsort函數(shù)對(duì)自定義結(jié)構(gòu)體數(shù)組進(jìn)行排序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11
C++中std::ios_base::floatfield報(bào)錯(cuò)已解決
在C++編程中,設(shè)置浮點(diǎn)數(shù)輸出格式時(shí)可能遇到std::ios_base::floatfield錯(cuò)誤,解決方法包括使用正確的格式化標(biāo)志組合,避免沖突的格式化設(shè)置,以及檢查流狀態(tài)標(biāo)志是否正確,通過(guò)這些方法可以有效避免浮點(diǎn)數(shù)格式化錯(cuò)誤,并確保輸出精確2024-09-09

