C 清空输入缓冲区,以及fflush(stdin)的使用误区和解决方法
转载:https://blog.csdn.net/Veniversum/article/details/62048870
对C 语言初学者来说,fflush(stdin)函数被解释为会清空输入缓冲区的一个系统函数,这是一个曾经几乎对过一半的说法,随着计算机科学的进步,在学习的过程中的逐步完善,将fflush(stdin)函数的过去与现在分析一下。
Personal thinking:
fflush(stdin) 会清空输入缓冲区中的内容,读取时输入缓冲区中的内容会被scanf函数逐个取走,正常case下scanf()函数可以根据返回值判断成功取走的数目;当发生异常读取的时候,如应该读取一个整形,结果输入缓冲区内当前的内容是个字符串,发生读取异常。发生读取异常之后,输入缓冲区中的内容并未被取走,那么下次循环之时,scanf()函数发现输入缓冲区中有内容(显然编译器不会关心这个内容是不是合法),于是不再等待user输入,直接尝试读取输入缓冲区中的内容,显而易见的又是一次读取异常,如此反复。
include <stdio.h>
int main( void )
{
int val,ret;
while(fflush(stdin),(ret = scanf("%d",&val)) != EOF)
printf(“%d\n”, val);
return 0;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
按照直觉,这个程序使用了fflush(stdin)以确保读取异常时不干涉下一次的读取。那么问题来了,事实果真如此么?
fflush(stdin)的前世今生
由Microsoft官方提供的MSDN 文档里也清楚地写着:fflush on input stream is an extension to the C standard(fflush 操作输入流是对 C 标准的扩充,注意啊,是Extension!)。即C 标准中根本没有定义 fflush(stdin),最新的C 11直接删去了曾经打擦边球的fflush(stdin)。
这也就是开头说的fflush(stdin)曾经几乎对过一半的原因:fflush()确实也是扩展,这就是几乎对过的意思,在vs 2013之前的版本里,包括 vc++ 6.0 fflush(stdin)也确实管用。现在即使在vs 2015环境下fflush(stdin)也不再起作用,遑论lunix系统下呢,这就是曾经对一半。
当然,如果毫不在乎程序的移植性,在可用 fflush(stdin)的版本里面这么写也没什么大问题。
清空输入缓冲区的可替代方法
既然fflush(stdin)现在在vs 2015 下不干活了,那么总得有接替背锅的角色,实现清空缓冲区的角色,下面根据查阅的结果,给出两种在C 可以实现清空输入缓冲区功能的可行方案。
首先声明下,使用setbuf(stdin,NULL)是GCC下可用的一种方法,但是没有解决掉缓存的问题,然而这里不予深究。
在vs 2015 下,可以用下面两种方法代替fflush(stdin)实现功能:
1)使用函数rewind(stdin)
从函数名上来看,这个函数应该是重定义了输入缓冲区的Location or Size,用这个方法得到新缓冲区,前后两个输入缓冲区并不是一样的(纯猜测,轻点打脸!),其实这个函数好像也是非标准定义的(不是很确定,因为在 C 11与 C 99 的 更新里面真没看到这个,不过也可以作为暂时管用的半个。都说到这里,多说一句,C 11 标准确实地删除了gets()函数,用gets_s()代替,关于这一点放在最后一个链接内)
2)使用scanf(“%*[^\n]%*c”),原理是用扫描集将缓冲区中的字符全部读取来实现清除输入缓冲区的动作,就效果来说非常管用,而且还跨平台。
乍一看这东西还有点深奥,给出详细的解释,也可以参考链接里面的解释说明。
对scanf(“%*[^\n]%*c”)解释:
%〔^\n〕将逐个读取缓冲区中的’\n’字符之前的其它字符,%后面的表示将读取的这些字符丢弃,前遇到’\n’字符时便停止读取操作,此时,缓冲区中尚有一个’\n’字符遗留,所以后面的%*c将读取并丢弃这个遗留的换行符,这里的星号和前面的星号作用相同。由于所有从键盘的输入都是以回车结束的,而回车会产生一个’\n’字符,所以将’\n’连同它之前的字符全部读取并丢弃之后,也就相当于清除了输入缓冲区。
关于fflush(stdout)
如果不考虑fflush(stdin)这个坑的话,它的兄弟fflush(stdout)还是有很大的作用的,简言之,fflush(stdout)强制输出当前输出缓冲区中的内容,一些在Debug下一些莫名其妙的error可以用fflush(stdout)立即输出在处理过程中的中间结果来确定error所在。
在查阅过程中发现一句话:
fflush(stdin)对输入流的操作是未定义的,所以这个还是要慎用,或许有副作用。
fflush(stdout)只是将需要输出的输出缓冲区中内容当即输出,利于调试且没有什么不良后果。
不管怎么说,向当年的二极管一样,fflush(stdin)也在计算机的发展过程中广泛使用过,时过境迁。一回首已百年身,也许以后的学习过程中再也不会遇见fflush(stdin)了。
参考资料:
1.ISO/IEC 9899:1999 (E) Programming languages— C 7.19.5.2 The fflush function
2.The C Programming Language 2nd Edition By Kernighan & Ritchie
相关链接:
scanf(“%[^\n]%*c”)怎么理解
scanf与缓存
C语言格式输入函数scanf()详解
用rewind(stdin)代替fflush(stdin)
C语言标准中的C99与最新的C11
C 清空输入缓冲区,以及fflush(stdin)的使用误区和解决方法的更多相关文章
- C语言清空输入缓冲区的N种方法对比
转自C语言清空输入缓冲区的N种方法对比 C语言中有几个基本输入函数: //获取字符系列 int fgetc(FILE *stream); int getc(FILE *stream); int get ...
- C语言清空输入缓冲区的N种方法对比【转】
转自:http://www.cnblogs.com/codingmylife/archive/2010/04/18/1714954.html C语言中有几个基本输入函数: //获取字符系列 int f ...
- C语言清空输入缓冲区的N种方法对比(转)
C语言中有几个基本输入函数: //获取字符系列 int fgetc(FILE *stream); int getc(FILE *stream); int getchar(void); //获取行系列 ...
- c++中清空输入缓冲区的方法(做cf的时候炸了)
C/C++ 四种清空输入缓冲区的方法 比较实用的一种 char c; while(c=getchar()!='\n'); 或者是这种 cin.ignore(count,c); count代表要清除的字 ...
- vs2019清空输入缓冲区
发现用cin.sync()在vs2019中不能清空输入缓冲区,以前的vs版本没试过,我看别人在vc中用cin.sync()可以清除,估计是IDE的问题..以下是我学习C++四个多月写的一整段代码 运行 ...
- 解压tar.gz文件报错gzip: stdin: not in gzip format解决方法
解压tar.gz文件报错gzip: stdin: not in gzip format解决方法 在解压tar.gz文件的时候报错 1 2 3 4 5 [Sun@localhost Downloads] ...
- C语言清空输入缓冲区
来源:http://blog.csdn.net/guanyasu/article/details/53153705 https://zhidao.baidu.com/question/5241738. ...
- [转][修]C清空输入缓冲区
为何要清空输入缓存区 读取时输入缓冲区中的内容会被scanf函数逐个取走,正常case下scanf()函数可以根据返回值判断成功取走的数目:但当发生读取异常之后,输入缓冲区中的内容并未被取走, ...
- dedecms清空栏目后,新建ID不从1开始的解决方法
在后台SQL运行器运行下面的语句,这样新建的栏目ID就从1开始了: ALTER TABLE `dede_arctype` AUTO_INCREMENT =1; (注意表名) 下面是文章的,运行后,发布 ...
随机推荐
- 极简 Node.js 入门 - 3.5 文件夹操作
极简 Node.js 入门系列教程:https://www.yuque.com/sunluyong/node 本文更佳阅读体验:https://www.yuque.com/sunluyong/node ...
- Laravel Pipeline原理及使用
Laravel Pipeline原理及使用 开发中可能遇到非常冗长的逻辑,以至于我们想将针对性逻辑拆分出来,但是又拿不准该如何拆分才能实现较高的扩展性并保证较高的维护性,或者说不知道如何优雅的将待处理 ...
- linux的五种IO模型
概念: 同步.异步.阻塞.非阻塞的概念 同步:所谓同步,发起一个功能调用的时候,在没有得到结果之前,该调用不返回,也就是必须一件事一件事的做,等前一件做完了,才能做下一件. 提交请求->等待服务 ...
- Agumaster 改善了pagination
页面越来越完善了.
- linux 安装ifconfig
一:使用yum命令下载安装wget 查看镜像中ifconfig安装包 yum search ifconfig yum install net-tools.x86_64 -y 下面按照提示一步步安装即可 ...
- Windows安装tensorflow教程 GPU版
PS:这是GPU版本,CPU版会用笔记本环境另写一篇博客. 前置准备 查看GPU型号 电脑桌面->右键我的电脑->选择管理->点击设备管理器 如下图: 如果不是英伟达显卡,那么不用 ...
- ckeditor4.0以上使用行间距插件lineheight报错修改
①从百度上下载一个 ckeditor 行距包,解压放到ckeditor/plugins目录下. ②在config.js 中添加 config.extraPlugins += (config.extra ...
- python基础二(list,tuple元祖、dic字典,字符串)
一.列表list 1.list定义 列表即数组 ,list或array..列表中的每个元素都有自己的编号,从0开始,编号也可叫做下标,角标,索引.最后一个元素的下标也可用-1表示.: list定义时, ...
- Redis 6.0.8 紧急发布,请尽快升级!
大家周末愉快啊,今天分享一则重要通知. Redis 6.0.8 于 2020/9/10 日晚紧急发布!!! 可以看到这是一个紧急更新版本,使用了 Redis 6.0.7 Sentinel(哨兵)以及 ...
- [Java并发编程之美]第1章 线程基础 补充知识
1.2线程创建与运行 创建线程有三种方式: 继承Thread类并重写run方法: 实现Runnable接口的run方法,new Thread时将该类对象作为参数传入: 实现Callable接口的cal ...