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

posted @ 2020-07-08 09:26  梅长苏枫笑  阅读(3052)  评论(0编辑  收藏  举报