C與C++中標準輸入實現方式上的一點區別

剛開始學C語言時遇到一個問題,如下代碼:

#include 
int main()
...{
char a,b;
printf("Please input the first character:");
scanf("%c", &a);
printf("Please input the second character:");
scanf("%c", &b);
printf("The two characters are %c, %c", a, b);
return 0;
}

程序運行結果如下:

圖1

而下面這段程序卻可以正常運行:

#include 
int main()
...{
char a;
int b;
printf("Please input the first character:");
scanf("%c", &a);
printf("Please input the second integer:");
scanf("%d", &b);
printf("The two characters are %c, %d", a, b);
return 0;
}

運行結果如下:

圖2

當時對這個問題很困惑,時間一長就慢慢淡忘了。

昨天一個同學問到一個C++運算符重載的問題,如下代碼,其中的while循環起什麼作用,好像註釋掉後程序運行沒什麼差別:

iostream& operator >> (iostream& is, b& s)
...{
char temp[b::MAX] = ...{0};
is.get();
is.get(temp, b::MAX);
if(strlen(temp)>0)
...{
strncpy(b.str, temp, b::MAX);
}
while(is && is.get()!=' ')
...{
continue;
}
return is;
}

其中,b爲一個自定義的類型,其中含有一個 char str[b::MAX] 的私有成員,b::MAX爲一個代表字符串長度的宏定義。

可見,當通過>>輸入字符串的長度超過b::MAX長度時,while循環就起作用了,它會丟棄超過b::MAX長度的輸入緩衝區的內容,直到換行符出現爲止,實際上就是一個清緩衝區的功能,以待下一次重新輸入字符串。爲了驗證我的分析,做了如下測試:

b b1, b2;
cin >> b1;
cin >> b2;
b1.show();
b2.show();

果然,當輸入字符串小於b::MAX時一切正常,當超過b::MAX時,當輸入完第一個字符串回車後,沒等輸入第二個字符串,程序就把b1和b2的內容都顯示出來了,而b2 的內容正是輸入的字符串超過b::MAX長度的部分,說明前面的分析是正確的。

今天又想起這個問題,突然聯想到以前C語言的兩個scanf連用產生的問題,靈機一動,難道也是輸入緩衝區沒有自動清空產生的問題?當連續用scanf輸入兩個字符的時候,輸入第一個字符後回車,結果換行符被當作第二個字符輸入了,scanf可不會區分什麼換行符還是普通字符,而當輸入一個字符,再輸入一個整數時,因爲類型不匹配的原因,scanf會將換行符忽略而等待輸入正確的整數後才返回,所以得到了正確的結果,哈哈,原來如此!!!

不禁又想,C++中引入了新的iostream流庫,它又是怎樣處理這個問題的呢?於是做了下面的實驗:

#include 
using namespace std;
int main()
...{
char a, b;
cout << "Please input the first character:";
cin >> a;
cout << "Please input the second character:";
cin >> b;
cout << a << "," << b;
return 0;
}

當輸入a回車,輸入b回車時,運行結果如下:

圖3

當連續輸入多個字符時,運行結果如下:

圖4

可見,C++中的標準輸入流已經對於換行符做了處理,看來在C++中利用標準輸入處理特殊字符要自己做一些工作了。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章