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