今天在C/C++交流学习群中,看到有人问了一个问题,总结如下。
首先,先来看出问题的代码:
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
int main() {
char ch;
while (scanf("%c", &ch) != EOF) {
switch (ch) {
case 'A': {
printf("90 ~ 100\n");
break;
}
case 'B': {
printf("80 ~ 89\n");
break;
}
case 'C': {
printf("70 ~ 79\n");
break;
}
case 'D': {
printf("60 ~ 69\n");
break;
}
case 'E': {
printf("0 ~ 59\n");
break;
}
default: {
printf("error\n");
break;
}
}
}
return 0;
}
我们再来看一下运行结果:
从上述结果可以看出,我们输入一个字母就会输出两个结果,一个正常结果,一个error,这是为什么呢?
- 这是因为输入缓冲区的问题,scanf函数功能是从输入缓冲区中读取一个指定大小的数据,由于这里是"%c",所以scanf每次从输入缓冲区中读取一个字节;
- 而我们每次输入的数据,表面上看是一个字母,其实不然,除了字母,我们每次还会输入一个回车’\n’;
- 就像上图演示的一样,我们从键盘上输入’A’,这时候输入缓冲区中有两个字节的数据,一个’A’和一个’\n’,所以scanf在第一次循环中会将’A’读取出来赋给ch变量,所以打印结果为"90 ~ 100",而第二次循环,scanf会将’\n’赋给变量ch,而’\n’的ASCII码值是10,所以这次会进入default分支,所以打印出的是"error"。
解决方法也很简单:
- 既然是缓冲区中的数据对结果有影响,那么我们每次循环结束的时候,都清空一下输入缓冲区就可以了。
改进代码如下:
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
int main() {
char ch;
while (scanf("%c", &ch) != EOF) {
switch (ch) {
case 'A': {
printf("90 ~ 100\n");
break;
}
case 'B': {
printf("80 ~ 89\n");
break;
}
case 'C': {
printf("70 ~ 79\n");
break;
}
case 'D': {
printf("60 ~ 69\n");
break;
}
case 'E': {
printf("0 ~ 59\n");
break;
}
default: {
printf("error\n");
break;
}
}
/*---清空输入缓冲区---*/
fflush(stdin);
}
return 0;
}