开启辅助访问 切换到宽版

精易论坛

 找回密码
 注册

QQ登录

只需一步,快速开始

用微信号发送消息登录论坛

新人指南 邀请好友注册 - 我关注人的新帖 教你赚取精币 - 每日签到


求职/招聘- 论坛接单- 开发者大厅

论坛版规 总版规 - 建议/投诉 - 应聘版主 - 精华帖总集 积分说明 - 禁言标准 - 有奖举报

查看: 1539|回复: 0
收起左侧

[其它] C/C++ 误区二:fflush(stdin)

[复制链接]
结帖率:50% (2/4)
发表于 2012-8-17 21:56:06 | 显示全部楼层 |阅读模式   湖南省长沙市
1.为什么 fflush(stdin)是错的
首先请看以下程序:
include <stdio.h>
int main( void )
{
int i;
for ( ;;) {
fputs("Please input an integer: ", stdout);
scanf("%d", &i);
printf("%d\n", i);
}
return 0;
}
这个程序首先会提示用户输入一个整数,然后等 待用户输入,如果用户输入的是整数,程序会输出刚才输入的整数,并且再次提 示用户输入一个整数,然后等待用户输入。但是一旦用户输入的不是整数(如小 数或者字母),假设 scanf 函数最后一次得到的整数是 2 ,那么程序会不停地 输出“Please input an integer: 2”。这是因为 scanf ("%d", &i); 只能接受整数,如果用户输入了字母,则这个 字母会遗留在“输入缓冲区”中。因为缓冲中有数据,故而 scanf 函数不会等待用户输入,直接就去缓冲中读取,可是缓冲中的却是字母,这个字 母再次被遗留在缓冲中,如此反复,从而导致不停地输出“Please input an integer: 2”。
也许有人会说:“居然这样,那么在 scanf 函数后面加上‘fflush(stdin);’,把输入缓冲清空掉不 就行了?”然而这是错的!C和C++的标准里从来没有定义过 fflush (stdin)。也许有人会说:“可是我用 fflush(stdin) 解决了这个问 题,你怎么能说是错的呢?”的确,某些编译器(如VC6)支持用 fflush (stdin) 来清空输入缓冲,但是并非所有编译器都要支持这个功能(linux 下 的 gcc 就不支持),因为标准中根本没有定义 fflush(stdin)。MSDN 文档里 也清楚地写着fflush on input stream is an extension to the C standard( fflush 操作输入流是对 C 标准的扩充)。当然,如果你毫不在乎程序的移植性 ,用 fflush(stdin) 也没什么大问题。以下是 C99 对 fflush 函数的定义:
int fflush(FILE *stream);
如果 stream 指向输出流或者更 新流(update stream),并且这个更新流最近执行的操作不是输入,那么 fflush 函数将把这个流中任何待写数据传送至宿主环境(host environment) 写入文件。否则,它的行为是未定义的。
原文如下:
int fflush (FILE *stream);
If stream points to an output stream or an update stream in which the most recent operation was not input, the fflush function causes any unwritten data for that stream to be delivered to the host environment to be written to the file; otherwise, the behavior is undefined.
其中,宿主环境可以理解为 操作系统或内核等。
由此可知,如果 stream 指向输入流(如 stdin) ,那么 fflush 函数的行为是不确定的。故而使用 fflush(stdin)是不正确的 ,至少是移植性不好的。

2. 清空输入缓冲区的方法
虽然不可以用 fflush(stdin),但是我 们可以自己写代码来清空输入缓冲区。只需要在 scanf 函数后面加上几句简单 的代码就可以了。
/* C 版本 */
#include <stdio.h>

int main( void )
{
int i, c;
for ( ; ; )
{
fputs("Please input an integer: ", stdout);
scanf("%d", &i);
if ( feof(stdin) || ferror(stdin) )
{ /* 如果用户输入文件结束标志(或 文件已被读完), */
/* 或者发生读写错误,则退出循环 */
/* do something */
break;
}
/* 没有发生错误,清空输入流。 */
/* 通过 while 循环把输入流中的余留数据“吃”掉 */
while ( (c = getchar()) != '\n' && c != EOF ) ;
/* 使用 scanf("%*[^\n]"); 也可以清空输入流, */
/* 不过会残留 \n 字符。*/
printf("%d\n", i);
}
return 0;
}

/* C++ 版本 */
#include <iostream>
#include <limits> // 为了 使用numeric_limits

using std::cout;
using std::endl;
using std::cin;
using std::numeric_limits;
using std::streamsize;

int main()
{
int value;
for ( ; ; )
{
cout << "Enter an integer: ";
cin >> value;
if ( cin.eof() || cin.bad() )
{ // 如果用户输入文件结束标志(或文件已被读完),
// 或者发生读写错误,则退出循环
// do something
break;
}
// 读到非法字符后,输入流将处于出错状态,
// 为了继续获取输入,首先要调用 clear 函数
// 来清除输入流的错误标记 ,然后才能调用
// ignore 函数来清除输入流中的数据。
cin.clear();
// numeric_limits<streamsize>::max() 返回输 入缓冲的大小。
// ignore 函数在此将把输入流中的数据清空。
// 这两个函数的具体用法请读者自行查询。
cin.ignore( numeric_limits<streamsize>::max(), '\n' );
cout << value << '\n';
}
return 0;
}
您需要登录后才可以回帖 登录 | 注册

本版积分规则 致发广告者

发布主题 收藏帖子 返回列表

sitemap| 易语言源码| 易语言教程| 易语言论坛| 易语言模块| 手机版| 广告投放| 精易论坛
拒绝任何人以任何形式在本论坛发表与中华人民共和国法律相抵触的言论,本站内容均为会员发表,并不代表精易立场!
论坛帖子内容仅用于技术交流学习和研究的目的,严禁用于非法目的,否则造成一切后果自负!如帖子内容侵害到你的权益,请联系我们!
防范网络诈骗,远离网络犯罪 违法和不良信息举报电话0663-3422125,QQ: 793400750,邮箱:wp@125.la
Powered by Discuz! X3.4 揭阳市揭东区精易科技有限公司 ( 粤ICP备12094385号-1) 粤公网安备 44522102000125 增值电信业务经营许可证 粤B2-20192173

快速回复 返回顶部 返回列表