编写C语言控制台程序时,如果使用scanf()函数作为输入,往往需要解决非法输入的问题。比如以下情况:
printf("输入一个0~9的整数a\n");
int a;
scanf("%d",&a);
此时可能出现如下几种典型的非法输入
- 123
- 123abc
- abc123
对于第一种,使用if判断a的范围是否满足要求即可。第二种和第三种则较为复杂,因为其中涉及到了字符的输入。第二种情况,scanf()会将123存入a,而abc留在了输入缓冲区;第三种情况,scanf()不会读取任何数据,abc123都留在了输入缓冲区。
如果只是判断这一次输入的合法性,可以利用scanf()的返回值。如果输入有效,scanf()会返回1,否则返回0。由此可以排除scanf()无法读取的输入。
然而,仅仅这样做是不够的。
假设之后继续运行如下代码:
printf("输入一个整数b\n");
int b;
scanf("%d",&b);
如果在输入a时,非法输入的abc或abc123留在了输入缓冲区,那么当scanf()再次执行时,并不会直接读取你第二次输入的数据,而是从上一次输入在输入缓冲区留下的数据开始。即使你在输入b时给出了合法的输入,也不会被正确读取。同理,使用getchar()得到的也是缓冲区中的第一个字符。
所以,问题的本质是:如果有多次输入,需要在每次输入之前,清除输入缓冲区。否则,之前由于非法输入导致留在输入缓冲区中的数据,会直接影响到这次输入。
而解决办法,通过查找各类论坛,总结出来有如下几种:
- setbuf(stdin,NULL);
- fflush(stdin);
- scanf(“%*[^\n]%*c”)
- rewind(stdin);
其原理各不相同,但经过测试,像
fflush(stdin);
这样的函数,在VC中可以清除输入缓冲区,但在XCode或gcc中不起作用。而
scanf(“%*[^\n]%*c”)
原理是读取输入缓冲区的所有字符,达到清除输入缓冲区的目的;这似乎也不能解决某些形式的非法输入。
最为有效的办法是在每次输入前加一句
rewind(stdin);
效果是坠吼的。