昨天在写其他代码时,突然使用了一下scanf函数,结果把自己用死了。
int a;
int b;
char op;
printf("输入两个数:");
scanf("%d%d", &a,&b);
printf("输入操作方法:");
scanf("%c", &op);
printf(" %d %c %d = ", a, op, b);
我的输入格式:
1 2
当我输完两个操作数,按下回车键,准备输入操作方法程序却变成了这。
这就是典型的scanf缓冲区问题。
当输入这句时1 2,输入缓冲区里存放着12\n,第一次scanf读走1 给 a,2 给 b,中间的空格丢弃,此时scanf已经读完。但是缓冲区里还遗留有一个 \n .当第二个scanf读入时首先检查缓冲区,发现缓冲区里还有 \n ,而且正好匹配%c,于是直接读走 \n 给 c。
这个问题我一开始就想到了缓冲区的问题,但是另一个问题又把我坑进去了。
我用 fflush(stdin);去刷新缓冲区。修改代码如下:
int a;
int b;
char op;
printf("输入两个数:");
scanf("%d%d", &a, &b);
printf("输入操作方法:");
fflush(stdin);
scanf("%c", &op);
printf(" %d %c %d = \n", a, op, b);
这样我以为解决了缓冲区的问题,殊不知完全不是我想的那样。于是我又查阅了fflush
> int fflush(FILE *stream);
如果stream指向输出流或者更新流(update stream),并且这个更新流最近执行的操作不是输入,那么fflush函数将把任何未被写入的数据写入stream指向的文件(如标准输出文件stdout)。否则,fflush函数的行为是不确定的。fflush(NULL)清空所有输出流和上面提到的更新流。如果发生写错误,fflush函数会给那些流打上错误标记,并且返回EOF,否则返回0。
我们不能用这种不确定的代码再我们的程序中。
正确解决缓冲区问题的方法
- 用rewind函数刷新缓冲区。
rewind(stdin);
可以达到我们的预期。
rewind函数是把指定流的读写指针重新指向开头。
2.scanf("%*c"); 读取缓冲区中指定格式的内容并且丢弃。
把缓冲区的\n读走并且丢弃。很明显可以达到预期
3.while(getchar() != "\n" && getchar() != EOF);
看了很多文章都说的用fflush(stdin)刷新输入缓冲区,其实是错误的。
我也是“实践出真知”,所以学知识一定要“求甚解”。
当然scanf还有很多奥妙的用法,很多文章都有介绍这里就不赘述了。