初学UNIX C时的一点小收获 (转)

初学UNIX C时的一点小收获 (转)[@more@]

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

请在vi编辑器下编写这段代码.

/*a.c*/

#include

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

跳出

XML:namespace prefix = o ns = "urn:schemas-microsoft-com:Office:office" />

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

main()

{

  int i = 0;

   int j = 0;

  char buffer[256];

  while( i != 1 )

  {

  fgets( buffer, 256, stdin );

  printf( "****** %s ******** ", buffer );

  fflush( stdin );

  j = scanf( "%d",  &i );

   printf( "%d ",  j );

  }

编译并执行,先输入任意一个值,比如a;这时候打印出来的输入缓冲区结果是a;然后再输入一个数字,比如2,这时候scanf将返回一个1,以示输入转换成功,同时输入缓冲区将会被清理,所以没有任何值输出。

接下来我们再试着输入一个a,回车,这时候scanf将返回一个0,以示输入转换出错,另外缓冲区中将会打印出a及回车,由此可见在接下来如果没有对缓冲区进行fflush的话,scanf将会接受到这个在输入缓冲区的a和回车,从而进入死循环!

也就是说刚才我们进行的输入,在输入了字母然后回车,scanf返回了一个EOF,而这个结果将会保留在缓冲区中,当下一次程序调用scanf时,由于这个输入缓冲区中所记忆的错误,scanf将不会正确执行,从而致使I的值无法改变,从而程序成了一个死循环!

问题找到了,解决的方法就是在调用scanf之前清除缓冲区,即在其前面加上:

fflush( stdin );

/*a.c*/

#include

main()

{

  int I = 0;

  while(I != 1)

  {

  fflush( stdin );

  scanf(“%d”, &I);

}

}

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


来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/10752043/viewspace-963478/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/10752043/viewspace-963478/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值