初学UNIX C时的一点小收获

王朝system·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

下面是我以前写代码时遇到的一个小程序,问题不大,但是忙活了好半天!下面我就用比较正是的方式来将这段经历送给大家。

请在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);

}

}

这下再一次测试这段代码!再接下来该干什么干什么吧!

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有  導航