标题:[求助]这个程序我那里搞错了?
只看楼主
lg_mic
Rank: 1
等 级:新手上路
帖 子:55
专家分:0
注 册:2007-9-18
 问题点数:0 回复次数:4 
[求助]这个程序我那里搞错了?
一个计算器程序,以下是代码

#include<stdio.h>
#include<conio.h>
int main(void)
{
int result,n1,n2,n,flag;
char c;
do{
flag=1;
printf("\nChoose an operator:1.+ 2.- 3.* 4./:\n");
scanf("%d",&n);
printf("Enter two numbers:");
scanf("%d,%d",&n1,&n2);
switch(n)
{
case 1: result=n1+n2; break;
case 2: result=n1-n2; break;
case 3: result=n1*n2; break;
case 4: result=n1/n2; break;
default: printf("Bad operator!\n");
flag=0;
}
if(flag)
printf("The result is %d\n",result);
printf("Would you like to do another problem?(Y/N):\n");
c=getch();
}while(c=='y'||c=='Y');
return 0;
}

运行后在红色的代码那里出问题了。
显示:
Would you like to do another problem?(Y/N):
(输入y)
Choose an operator:1.+ 2.- 3.* 4./:
y
为什么会显示出上面的那个y?getch()不是不显示的么?

还有就是如果最开始运行的时候输入错误的数据的话也会出问题
显示:
Choose an operator:1.+ 2.- 3.* 4./:
(输入a)
Enter two numbers:
Bad operator!
Would you like to do another problem?(Y/N):
(输入y)
Choose an operator:1.+ 2.- 3.* 4./:
Enter two numbers:
Bad operator!
Would you like to do another problem?(Y/N):
(按任意键退出,输入y的话继续循环)

为什么会出现这样的问题?

搜索更多相关主题的帖子: result 计算器 case 
2007-09-20 11:40
josen0205
Rank: 2
来 自:江苏
等 级:论坛游民
帖 子:307
专家分:52
注 册:2007-5-8
得分:0 
#include<stdio.h>
#include<conio.h>
int main()
{
int result,n1,n2,n;
char c;
while(1)
{
printf("\nChoose an operator:1.+ 2.- 3.* 4./:");
if(scanf("%d",&n)!=1||n<1||n>4)
{
printf("Bad operator!\n");
fflush(stdin);
continue;
}
printf("Enter two numbers:");
scanf("%d,%d",&n1,&n2);
switch(n)
{
case 1: result=n1+n2; break;
case 2: result=n1-n2; break;
case 3: result=n1*n2; break;
case 4: result=n1/n2; break;
default :
break;
}
printf("The result is %d\n",result);
printf("Would you like to do another problem?(Y/N):");
c=getch();
putchar(c);
if(c=='n'||c=='N')
break;
}
getch();
return 0;
}

只有想不到,没有做不到
2007-09-20 14:11
lg_mic
Rank: 1
等 级:新手上路
帖 子:55
专家分:0
注 册:2007-9-18
得分:0 
哦,我想我懂了,我那个的缓冲区没有清空,所以它直接从缓冲区中读取数据,导致了不断的循环。
以下是我在其他地方找到的解释:

首先请看以下程序:

#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”。

另外,fflush(stdin);可以用么?我把那个全文转过来

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);,把输入缓冲清空掉不就行了?”然而这是错的!CC++标准里从来没有定义过 fflush(stdin)。也许有人会说:“可是我用 fflush(stdin) 解决了这个问题,你怎么能说是错的呢?”的确,某些编译器(如VC6)支持用 fflush(stdin) 来清空输入缓冲,但是并非所有编译器都要支持这个功能(linux 下的 gcc 不支持),因为标准中根本没有定义 fflush(stdin)MSDN 文档里也清楚地写着fflush on input stream is an extension to the C standardfflush 操作输入流是对 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;
}

参考资料

ISO/IEC 9899:1999 (E) Programming languages C 7.19.5.2 The fflush function

The C Programming Language 2nd Edition By Kernighan & Ritchie

ISO/IEC 14882(1998-9-01)Programming languages C++

本文版权归 蚂蚁的 C/C++ 标准编程 以及 作者 antigloss 共同所有,转载请注明原作者和出处。谢谢。


2007-09-20 14:40
百年不亮
Rank: 3Rank: 3
等 级:新手上路
威 望:8
帖 子:789
专家分:0
注 册:2006-4-14
得分:0 
很多人会遇到这种问题。

要不楼主写一个介绍"输入缓冲区垃圾及处理"的FAQ吧,内容写详细些,排版做漂亮点,我会加精后加入FAQ的。
2007-09-20 14:52
lg_mic
Rank: 1
等 级:新手上路
帖 子:55
专家分:0
注 册:2007-9-18
得分:0 
恩,等我晚上有时间的,把我遇到的这几个情况归纳一下写出来。

2007-09-20 14:55



参与讨论请移步原网站贴子:https://bbs.bccn.net/thread-171454-1-1.html




关于我们 | 广告合作 | 编程中国 | 清除Cookies | TOP | 手机版

编程中国 版权所有,并保留所有权利。
Powered by Discuz, Processed in 0.157826 second(s), 8 queries.
Copyright©2004-2025, BCCN.NET, All Rights Reserved