c++中cin/cout與scanf/printf的區(qū)別比較
cin 、cout
基本說(shuō)明:
cin代表標(biāo)準(zhǔn)輸入設(shè)備,使用提取運(yùn)算符 ">>" 從設(shè)備鍵盤取得數(shù)據(jù),送到輸入流對(duì)象cin中,然后送到內(nèi)存。
cin是輸入流,cout是輸出流,重載了">>"、"<<"運(yùn)算符,包含在頭文件<iostream>中。
先把要輸出的東西存入緩沖區(qū),再輸出,導(dǎo)致效率降低,cin是自動(dòng)判斷你的變量類型,比如一個(gè)char數(shù)據(jù)只能用默認(rèn)的char方法取數(shù)據(jù)。
scanf 、printf
基本說(shuō)明:
scanf是格式化輸入,printf是格式化輸出,包含在頭文件<stdio.h>中。
因?yàn)閟canf是用指針操作的,沒(méi)有類型安全機(jī)制,比如一個(gè)char類型數(shù)據(jù)你就可以用%f獲得輸入,而不會(huì)報(bào)錯(cuò),但在運(yùn)行時(shí)會(huì)出現(xiàn)異常。
scanf()函數(shù)取數(shù)據(jù)是遇到回車、空格、TAB就會(huì)停止,如例1,第一個(gè)scanf()會(huì)取出"Hello",而"world!"還在緩沖區(qū)中,這樣第二個(gè)scanf會(huì)直接取出這些數(shù)據(jù),而不會(huì)等待從終端輸入。
例1:
#include <stdio.h>
int main()
{
char str1[20], str2[20];
scanf("%s",str1);
printf("%s\n",str1);
scanf("%s",str2);
printf("%s\n",str2);
return 0;
}測(cè)試一輸入:
Hello world!
輸出:
Hello
world!
第一個(gè)scanf()會(huì)取出"Hello",而"world!"還在緩沖區(qū)中,這樣第二個(gè)scanf會(huì)直接取出這些數(shù)據(jù),而不會(huì)等待從終端輸入。
為了避免出現(xiàn)上述問(wèn)題,必須要清空緩沖區(qū)的殘留數(shù)據(jù),可以用以下的方法解決:
方法1:C語(yǔ)言里提供了函數(shù)清空緩沖區(qū),只要在讀數(shù)據(jù)之前先清空緩沖區(qū)就沒(méi)問(wèn)題了!
這個(gè)函數(shù)是fflush(stdin)。
方法2:自己取出緩沖區(qū)里的殘留數(shù)據(jù)。
(說(shuō)實(shí)話這個(gè)語(yǔ)句我也沒(méi)看懂,呵呵!為什么格式控制是這樣的!希望高手指點(diǎn)一下?。?br />scanf("%[^\n]",string);
gets()
基本說(shuō)明:
gets()函數(shù)用來(lái)從標(biāo)準(zhǔn)輸入設(shè)備(鍵盤)讀取字符串直到換行符結(jié)束,但換行符會(huì)被丟棄,然后在末尾添加'\0'字符。包含在頭文件<stdio.h>中。
gets(s)函數(shù)與 scanf("%s",&s) 相似,但不完全相同,使用scanf("%s",&s) 函數(shù)輸入字符串時(shí)存在一個(gè)問(wèn)題,就是如果輸入了空格會(huì)認(rèn)為字符串結(jié)束,空格后的字符將作為下一個(gè)輸入項(xiàng)處理,但gets()函數(shù)將接收輸入的整個(gè)字符串直到遇到換行為止。
原型:
char*gets(char*buffer);
例2:
#include <stdio.h>
int main()
{
char str1[20], str2[20];
gets(str1);
printf("%s\n",str1);
gets(str2);
printf("%s\n",str2);
return 0;
}測(cè)試:
Hello world! [輸入]
Hello world! [輸出]
12345 [輸入]
12345 [輸出]
為了避免出現(xiàn)上述問(wèn)題,必須要清空緩沖區(qū)的殘留數(shù)據(jù),可以用以下的方法解決:
方法1:C語(yǔ)言里提供了函數(shù)清空緩沖區(qū),只要在讀數(shù)據(jù)之前先清空緩沖區(qū)就沒(méi)問(wèn)題了!
這個(gè)函數(shù)是fflush(stdin)。
方法2:自己取出緩沖區(qū)里的殘留數(shù)據(jù)。
(說(shuō)實(shí)話這個(gè)語(yǔ)句我也沒(méi)看懂,呵呵!為什么格式控制是這樣的!希望高手指點(diǎn)一下!)
scanf("%[^\n]",string);
輸入操作的原理
與前一節(jié)中提到的scanf函數(shù)一樣,程序的輸入都建有一個(gè)緩沖區(qū),即輸入緩沖區(qū)。一次輸入過(guò)程是這樣的,當(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ǔ)句失效的原因!
cin的一些輸入函數(shù)和操作符
cin is a extern istream object。提供了很多可用的成員函數(shù)和重載的操作符,如:cin<<, cin.get(), cin.getline()等。下面我們來(lái)了解一下這幾個(gè)函數(shù):
一. cin<<
該操作符是根據(jù)后面變量的類型讀取數(shù)據(jù)。
輸入結(jié)束條件 :遇到Enter、Space、Tab鍵。
對(duì)結(jié)束符的處理 :丟棄緩沖區(qū)中使得輸入結(jié)束的結(jié)束符(Enter、Space、Tab)
二.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à)的。
讀取字符串的情況:
cin.get(array_name, Arsize)是用來(lái)讀取字符串的,可以接受空格字符,遇到Enter結(jié)束輸入,按照長(zhǎng)度(Arsize)讀取字符, 會(huì)丟棄最后的Enter字符。
程序6:
#include <iostream>
using namespace std;
int main ()
{
char a[20];
cin.get(a, 10);
cout<<a<<endl;
return 0;
}測(cè)試一輸入:
abc def[Enter]
輸出:
abc def
【分析】說(shuō)明該函數(shù)輸入字符串時(shí)可以接受空格。
測(cè)試二輸入:
1234567890[Enter]
輸出:
123456789
【分析】輸入超長(zhǎng),則按需要的長(zhǎng)度取數(shù)據(jù)。
程序7:
#include <iostream>
using namespace std;
int main ()
{
char ch, a[20];
cin.get(a, 5);
cin>>ch;
cout<<a<<endl;
cout<<(int)ch<<endl;
return 0;
}測(cè)試一輸入:
12345[Enter]
輸出:
1234
53
【分析】第一次輸入超長(zhǎng),字符串按長(zhǎng)度取了"1234",而'5'仍殘留在緩沖區(qū)中,所以第二次輸入字符沒(méi)有從鍵盤讀入,而是直接取了'5',所以打印的ASCII值是53('5'的ASCII值)。
測(cè)試二輸入:
1234[Enter]
a[Enter]
輸出:
1234
97
【分析】第二次輸入有效,說(shuō)明該函數(shù)把第一次輸入后的Enter丟棄了!
三.cin.getline()
cin.getline() 與 cin.get(array_name, Arsize)的讀取方式差不多,以Enter結(jié)束,可以接受空格字符。按照長(zhǎng)度(Arsize)讀取字符, 會(huì)丟棄最后的Enter字符。
但是這兩個(gè)函數(shù)是有區(qū)別的:
cin.get(array_name, Arsize)當(dāng)輸入的字符串超長(zhǎng)時(shí),不會(huì)引起cin函數(shù)的錯(cuò)誤,后面的cin操作會(huì)繼續(xù)執(zhí)行,只是直接從緩沖區(qū)中取數(shù)據(jù)。但是cin.getline()當(dāng)輸入超長(zhǎng)時(shí),會(huì)引起cin函數(shù)的錯(cuò)誤,后面的cin操作將不再執(zhí)行。(具體原因?qū)⒃谙乱徊糠?quot;cin的錯(cuò)誤處理"中詳細(xì)介紹)
程序8:
#include <iostream>
using namespace std;
int main ()
{
char ch, a[20];
cin.getline(a, 5);
cin>>ch;
cout<<a<<endl;
cout<<(int)ch<<endl;
return 0;
}測(cè)試輸入:
12345[Enter]
輸出:
1234
-52
【分析】與cin.get(array_name, Arsize)的例程比較會(huì)發(fā)現(xiàn),這里的ch并沒(méi)有讀取緩沖區(qū)中的5,而是返回了-52,這里其實(shí)cin>>ch語(yǔ)句沒(méi)有執(zhí)行,是因?yàn)閏in出錯(cuò)了!
相關(guān)文章
C++基于Boost.Asio實(shí)現(xiàn)端口映射器的過(guò)程詳解
Boost.Asio 是一個(gè)功能強(qiáng)大的 C++ 庫(kù),用于異步編程和網(wǎng)絡(luò)編程,它提供了跨平臺(tái)的異步 I/O 操作,在這篇文章中,我們將深入分析一個(gè)使用 Boost.Asio 實(shí)現(xiàn)的簡(jiǎn)單端口映射服務(wù)器,文中有詳細(xì)的代碼講解,需要的朋友可以參考下2023-11-11
OpenCV?直方圖均衡化的實(shí)現(xiàn)原理解析
直方圖均衡化是通過(guò)拉伸像素強(qiáng)度分布范圍來(lái)增強(qiáng)圖像對(duì)比度的一種方法,今天通過(guò)本文給大家介紹OpenCV?直方圖均衡化的實(shí)現(xiàn)原理解析,感興趣的朋友跟隨小編一起看看吧2022-01-01
C語(yǔ)言實(shí)現(xiàn)電腦關(guān)機(jī)程序
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)電腦關(guān)機(jī)程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-02-02
C++中的多態(tài)問(wèn)題—理解虛函數(shù)表及多態(tài)實(shí)現(xiàn)原理
這篇文章主要介紹了C++中的多態(tài)問(wèn)題—理解虛函數(shù)表及多態(tài)實(shí)現(xiàn)原理,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02

