转载——关于scanf造成死循环

很多人在使用scanf的时候都不会去检查它的返回值, 包括我在这之前也没有去注意它, 如果你方便去翻翻c语言的教科书, 估计也很难找到有检查scanf的例子或代码段, 至少我还没有见过, 当然, 书上的代码毕竟只是一个练习, 和真正用于产品级别的代码有很大的差别。

一个偶然的机会, 在独立编写一个10010客户服务端系统时,让我注意到检查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( "�", &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( "�", &iMaxDataNum );

}

其中while条件中的0 >= iMaxDataNum 是在判断输入值的范围问题,不在我们讨论范围。

后面的

if( 0 >= iReadRslt )

{

   while( '\n' != getchar() )

   {

    continue;

   }

}

就是用来解决问题的。当用户输入非法值时,用一个while循环,将本次输入的非法值连同最后确定输入时敲的回车,都从输入缓冲中读出来。这样就避免了上次的非法输入值被不断读入,从而造成的多次循环

----------------------------------------------------------------------------------------------    方案2:

直接使用缓冲区清除函数fflush(stdin); //stdin代表标准输入;

以下是一个例子:

#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;

   }

   func = -1;

}

}

原文地址:https://www.cnblogs.com/zengjianrong/p/3032748.html