刚开始学C语言时遇到一个问题,如下代码:
#include <stdio.h>
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;
}
程序运行结果如下:
而下面这段程序却可以正常运行:
#include <stdio.h>
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;
}
运行结果如下:
当时对这个问题很困惑,时间一长就慢慢淡忘了。
昨天一个同学问到一个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 <iostream>
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回车时,运行结果如下:
当连续输入多个字符时,运行结果如下:
可见,C++中的标准输入流已经对于换行符做了处理,看来在C++中利用标准输入处理特殊字符要自己做一些工作了