下面是我以前写代码时遇到的一个小程序,问题不大,但是忙活了好半天!下面我就用比较正是的方式来将这段经历送给大家。
请在vi编辑器下编写这段代码.
/*a.c*/
#include<stdio.h>
main()
{
int I = 0;
while(I != 1)
{
scanf(“%d”, &I);
}
}
我曾经问过几个朋友:这段代码有没有问题;都说:应该没问题吧!那么我们来测试一下。
我们先写一段简单的makefile
下面在UNIX下输入
$vi makefile
在vi编辑器下编辑下面这段代码
#这是一段makefile代码
#作用是:编译a.c成可执行文件a
HEADERS = include/stdio.h
SOURCES = a.c
PRODUCT = a
CC = cc
$(PRODUCT):$(SOURCE)
$(CC) -o $(PRODUCT) $(SOURCES)
编辑完成后,我们来执行makefile
$ make<回车>
cc -o a a.c
现在一个可执行的a已经生成了,下面开始写我们的单元测试计划
测试编 号
输入
输出
测试结果
分析
1
1<回车>
跳出
2
2<回车>
进入下一次输入状态
3
1.2<回车>
跳出
4
1.9<回车>
跳出
5
0.9<回车>
进入下一次输入状态
6
a<回车>
进入下一次输入状态
$ a
1
$
…
…
…
边测试边填写测试计划表
测试编 号
输入
输出
测试结果
分析
1
1<回车>
跳出
正确
2
2<回车>
进入下一次输入状态
正确
3
1.2<回车>
跳出
正确
4
1.9<回车>
跳出
正确
5
0.9<回车>
进入下一次输入状态
正确
6
a<回车>
进入下一次输入状态
看样子没问题,还有一步就完成了,输入一个非数值,a,回车!好像没问题!再输入1回车!!问题出现了!没有退出循环!Ctrl+c退出吧!
$ a.out
a
1
^C$
测试编 号
输入
输出
测试结果
分析
1
1<回车>
跳出
正确
此代码有一处严重错误将不予以发布
2
2<回车>
进入下一次输入状态
正确
3
1.2<回车>
跳出
正确
4
1.9<回车>
跳出
正确
5
0.9<回车>
进入下一次输入状态
正确
6
a<回车>
进入下一次输入状态
出错
为什么会这样呢?出现了严重的BUG,马上调试。首先让我们在scanf语句后面加上一句:
printf(“%d”, I),看一看究竟I后来发生了什么样的变化,然后重复编号6的测试步骤,你会发现程序执行进入了死循环!它在不停的打印0!也就是说scanf根本不执行了!为什么会这样呢?我们的scanf 去哪里了?经过一番认真学习和听取老师们的指导终于明白,原来是这样的。先看下面的概念。
scanf在发生输入错或者读到了EOF,scanf立即返回,如果它读入了一个不正确字符(如,在应是数字的地方出现了字母),它也立即返回。scanf返回它所完成的转换的数目;如果它没有完成任何转换,则它返回EOF。
scanf在读到一个换行符时才停止。除非用户清除换行符,否则它将保留在标准输入设备的缓冲区中。如果程序员在第二次调用scanf前没有清除输入缓冲区,则会产生错误的结果。
下面我们做个实验,将代码改写成以下。
#include<stdio.h>
main()
{
int i = 0;
int j = 0;
char buffer[256];
while( i != 1 )
{
fgets( buffer, 256, stdin );
printf( "******\n%s\n********\n", buffer );
fflush( stdin );
j = scanf( "%d", &i );
printf( "%d\n", j );
}
编译并执行,先输入任意一个值,比如a;这时候打印出来的输入缓冲区结果是a;然后再输入一个数字,比如2,这时候scanf将返回一个1,以示输入转换成功,同时输入缓冲区将会被清理,所以没有任何值输出。
接下来我们再试着输入一个a,回车,这时候scanf将返回一个0,以示输入转换出错,另外缓冲区中将会打印出a及回车,由此可见在接下来如果没有对缓冲区进行fflush的话,scanf将会接受到这个在输入缓冲区的a和回车,从而进入死循环!
也就是说刚才我们进行的输入,在输入了字母然后回车,scanf返回了一个EOF,而这个结果将会保留在缓冲区中,当下一次程序调用scanf时,由于这个输入缓冲区中所记忆的错误,scanf将不会正确执行,从而致使I的值无法改变,从而程序成了一个死循环!
问题找到了,解决的方法就是在调用scanf之前清除缓冲区,即在其前面加上:
fflush( stdin );
/*a.c*/
#include<stdio.h>
main()
{
int I = 0;
while(I != 1)
{
fflush( stdin );
scanf(“%d”, &I);
}
}
这下再一次测试这段代码!再接下来该干什么干什么吧!