| 网站首页 | 业界新闻 | 小组 | 威客 | 人才 | 下载频道 | 博客 | 代码贴 | 在线编程 | 编程论坛
欢迎加入我们,一同切磋技术
用户名:   
 
密 码:  
共有 2241 人关注过本帖, 1 人收藏
标题:关于while循环体内语句被漏执行造成死循环问题
只看楼主 加入收藏
凌彬严
Rank: 1
等 级:新手上路
帖 子:38
专家分:1
注 册:2009-7-13
结帖率:100%
收藏(1)
已结贴  问题点数:5 回复次数:6 
关于while循环体内语句被漏执行造成死循环问题
#include <stdio.h>

int main()
{
long a,n,i,sum1=0,sum2=0;

do
    {
    printf("请输入a,n:\n");
    scanf("%ld,%ld",&a,&n);/*当表达式为假时,再执行循环体语句时,该行语句无效了,直接不断的printf*/
    }
while(!(a>=0&&a<=9&&n>=0&&n<=9));

sum1=a;
sum2=a;
for(i=1;i<n;i++)
{
sum1=sum1*10+a;
sum2+=sum1;
}
printf("S=%ld\n",sum2);
getch();
}

当输入非法,如:a,3时,死循环就出现了,怎么回事?
搜索更多相关主题的帖子: while 死循环 
2009-08-15 12:55
godbless
Rank: 8Rank: 8
等 级:蝙蝠侠
威 望:1
帖 子:216
专家分:950
注 册:2009-7-24
收藏
得分:2 
scanf("%ld,%ld",&a,&n);后面加上下面这句

do
    {
    printf("请输入a,n:\n");
    scanf("%ld,%ld",&a,&n);
    while(getchar()!='\n'); /*当表达式为假时,再执行循环体语句时,该行语句无效了,直接不断的printf*/
    }
while(!(a>=0&&a<=9&&n>=0&&n<=9));
2009-08-15 13:07
西园竹
Rank: 5Rank: 5
等 级:职业侠客
帖 子:41
专家分:305
注 册:2009-8-8
收藏
得分:3 
scanf加一句fflush(stdin);清空输入缓存,避免scanf自动执行
2009-08-15 13:27
凌彬严
Rank: 1
等 级:新手上路
帖 子:38
专家分:1
注 册:2009-7-13
收藏
得分:0 
谢谢两位,能说说这个死循环是如何发生的吗?
2009-08-15 13:33
soky
Rank: 4
等 级:业余侠客
帖 子:126
专家分:228
注 册:2009-7-13
收藏
得分:0 
scanf("%ld,%ld",&a,&n);//你录入的是两个doublue 类型的,不是字符的
2009-08-15 15:33
wangyf
Rank: 2
等 级:论坛游民
帖 子:40
专家分:45
注 册:2009-6-27
收藏
得分:0 
scanf("%ld,%ld",&a,&n);/*当表达式为假时,再执行循环体语句时,该行语句无效了,直接不断的printf*/
再在后面加上getchar();//接收输入的回车
2009-08-15 15:36
凌彬严
Rank: 1
等 级:新手上路
帖 子:38
专家分:1
注 册:2009-7-13
收藏
得分:0 
终于找到满意的答案了,如下:
以下转自http://hi.baidu.com/tangzs/blog/item/4cb1304e96f9a6cad0c86ac6.html关键词:对scanf返回值的检查 / 及对输入缓冲的清空,scanf造成的死循环
很多人在使用scanf的时候都不会去检查它的返回值, 包括我在这之前也没有去注意它, 如果你方便去翻翻c语言的教科书, 估计也很难找到有检查scanf的例子或代码段, 至少我还没有见过, 当然, 书上的代码毕竟只是一个练习, 和真正用于产品级别的代码有很大的差别。一个偶然的机会, 让我注意到检查scanf返回值的重要性, 写下来和大家分享一下:)下面是一个最简单不过的scanf用法,但却有隐患:int n;scanf( "%d", &n);...scanf要求取一个整数,但如果输入的是一个字母,或其它不是数字的字符时,情况会怎么样呢? 结果是scanf不读取数据,不修改n的值直接返回了,这样,n就变成了一个未被初始化的变量,其值不可确定,因为其值不确定,所以就算你对n作是否合法的判断,也还是可能会有问题,你也许会说,用户应该会输入一个合法的值的,但是,有时scanf并不总是从键盘中读入数据,它可能从其它的设备中去读,例如管道,UNIX中允许这样的玩法。你也许还会说,情况没有那么糟啦,可以先初始化n为零,再去判断n是否合法嘛,例如下面这样:int n = 0;scanf( "%d", &n);if (n != 0)  
{    //OK,读到用户的值了}这样好像还是有麻烦啊,这个初始值你怎么定义才好? 0吗? 程序若是需要0的时候怎么办?让我们再看下面的代码,要求用户输入一个大于0的数为止:int n;for(;;) {    printf( "pls input:");    scanf( "%d", &n);    if (n > 0) {        break;    }}上面的代码中,当用户输入的不是一个整数时,会死循环,而不是用户想要的:当用户输入非法值时一直让用户输入一个合法值为止。为什么会这样呢,因为当scanf检查到所输入的是不合法的值时,它并不会读取它,因此数据会留在缓冲区中,当下一次调用scanf时,会去读缓冲区中的数据,因此仍然会读到一个非法的值----------------------------------------------------------------------------------------------
以下是比较好的解决方案:
方案1:
1. 用类似 iReadRslt = scanf( "%8d", &iMaxDataNum ); 的方法取得scanf的返回值并对其进行是否为0的判断,可以检查输入的合法性(比如要求输入整数,对方却输入字母,此时返回值为0)。2. 上面引文中提到的输入值不合法,同时不合法的输入值贮留输入缓冲区导致scanf不断读取非法值,程序死循环的问题,可以使用下面方法解决:while( ( 0 >= iMaxDataNum ) || ( 0 >= iReadRslt ) )
{
if( 0 >= iReadRslt )
{
   while( '\n' != getchar() )
   {
    continue;
   }
}
 
printf( "Invalid number, please input the correct one." );
printf( "\nHow many Tel No. can be saved!\n" );
iReadRslt = scanf( "%8d", &iMaxDataNum );
 
}其中while条件中的0 >= iMaxDataNum 是在判断输入值的范围问题,不在我们讨论范围。后面的if( 0 >= iReadRslt )
{
   while( '\n' != getchar() )
   {
    continue;
   }}
就是用来解决问题的。当用户输入非法值时,用一个while循环,将本次输入的非法值连同最后确定输入时敲的回车,都从输入缓冲中读出来。这样就避免了上次的非法输入值被不断读入,从而造成的多次循环----------------------------------------------------------------------------------------------------------------------
 
方案2:
直接使用缓冲区清除函数fflush(stdin); //stdin代表标准输入;
以下是一个例子:
 
 
/* 程序功能:解决scanf在非法输入后的死循环问题 */
#include <stdio.h>
#include <stdlib.h>
 
void main()
{  
int func = -1;   //初始化为非法标志-1;
 
while(1)
{
   printf ( "-----------------");
   printf ( "\n*** 主 菜 单 ***\n" );
   printf ( " 0. 退出\n");
   printf ( " 1. show hello\n");
   printf ( " 2. 清楚屏幕\n");
   printf ( "-----------------\n");  
   printf("选择(0--2):");  
 
   scanf("%d",&func);  
 
   if(getchar()!='\n')func=-1; //避免类似“1qweqweqw”这样的输入,即只能输入0或1或2;
 
   fflush(stdin); //清空输入缓冲区;
 
   switch ( func )
   {
    case 0:  
     exit(0);  
     break;
    case 1:
     printf ( "\nhello,world~~~\n\n");
     break;
    case 2:
     system("cls");
     break;
    default:
     printf("输入错误!\n");  
     break;
   } /* end switch */
 
   func = -1;
}
}
2009-08-15 19:17
快速回复:关于while循环体内语句被漏执行造成死循环问题
数据加载中...
 
   



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

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