初学UNIX C时的一点小收获

类别:编程语言 点击:0 评论:0 推荐:

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

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

}

}

 

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

本文地址:http://com.8s8s.com/it/it29068.htm