首页 > C/C++语言 > C/C++基本语法 > c源程序常见错误分析
2005
12-29

c源程序常见错误分析




一、C语言出错有两种情况:


1、语法错误。指编程时违背了C语法的规定,对这类错误,编译程序一般都能够给出“出错信息”,并且告诉在哪一行出错及出错的类型。只要仔细检查,是可以很快发现错误并排除的。

2、逻辑错误。程序并无违背语法规则,但程序执行结果与原意不符。这是由于程序设计人员写出的源程序与设计人员的本意不相同,即出现了逻辑上的混乱。

例如:

unsigned char i=1;

unsigned int sum=0;

while (i<=100)

    sum=sum+i;

    i++;

在上例中,设计者本意是想求从1到100的整数和,但是由于循环语句中漏掉了大括号,使循环变为死循环而不是求累加。对于这种错误,C编译通常都不会有出错信息(因为符合C语法,但有部分编译系统会提示有一个死循环)。对于这类逻辑错误,比语法错误更难查找,要求程序设计者有丰富的设计经验(不会有类似的错误)和有丰富的排错经验(通过仿真能够很快发现问题)。

二、初学者在编写C源程序时常见错误及分析

1、忘记定义变量就使用

例如:

main ()

{

x=3;

y=x;

}

在上式中看似正确,实际上却没有定义变量x和y的类型。C语言规定,所有的变量必须先定义,后使用。因此在函数开头必须有定义变量x和y的语句,应改为:

main ()

{

int x,y;

x=3;

y=x;

}

2、变量没有赋值初就直接使用。

例如:

unsigned int addition (unsigned int n)

{

unsigned int i;

unsigned int sum;

for (i=0;i<n;i++)

sum+=i;

return (sum);

}

上例中本意是计算1到n之间整数的累加和,但是由于sum没有赋初值,sum中的值是不确定的,因此得不到正确的结果。应改为如下:

unsigned int addition (unsigned int n)

{

unsigned int i;

unsigned int sum=0;

for (i=0;i<n;i++)

sum+=i;

return (sum);

}

或者将sum定义为全局变量(全局变量在初始化时自动赋值“0”)。

unsigned int sum;

unsigned int addition (unsigned int n)

{

unsigned int i;

for (i=0;i<n;i++)

sum+=i;

return (sum);

}

3、输入输出的数据类型与所用格式说明符不一致

例如:

main ( )

{

int a=3,b=4.5;

printf(“%f %d\n”,a,b);

}

在上例中,a与b变量错位,但编译时并不给出出错信息,输出结果为:

0.000000  16402

它们并不是按赋值的规则进行转换,如把3转换成3.0,把4.5转换成4,而是将存储单元中的数据按格式符的要求的宽度直接输出,如b占4个字节却用“%d”说明,则只有最后两个字节中的数据当成一个整数输出,a也相同,将a地址前两个字节(并不属于a)与变量a的两个字节当成一个4个字节的浮点数输出。

4、没有注意数据的数值范围

8位单片机适用的C编译器,对字符型变量分配一个字节,对整型变量分配二个字节,因此有数值范围的问题。有符号的字符变量的数值范围为-128~127,有符号的整型变量的数值范围为-32768~32767。其它类型变量的范围这里就不再一一列举,请读者参见相应编译器的使用手册。

例如:

main ()

{

char x;

x=300;

}

在上例中,有很多读者会认为x的值就是300,实际上却是错误的。

300的二进制为0b100101100,赋值给x时,将赋值最后的8位,高位截去,因此x的值实际上为0b101100(即整数44)。

如果将500赋给一个有符号的字符型变量时,变量内存储的值还会变成负数,由读者自行分析原因。

5、输入变量时忘记使用地址符号

常见是忘记使用地址符:

例如:

main ()

{

int a,b;

scanf (“%d%d”,a,b);

}

应改为:

scanf (“%d%d”,&a,&b);

6、输入时数组的组织方式与要求不符

scanf (“%d %d”,a,b);

如果输入数据格式为:

3,4

则是错误的,两个数据之间应用空格分来分隔,应为:

3 4

7、误把“=”作为关系运算符“等于”

在数学和其它高级语言中,都是把“=”作为关系运算符“等于”,因此容易将程序误写为:

if (a=b)

c=0;

else

   c=1;

在上例中,本意是如果a等于b,则c=0,否则c=1。但C编译系统却认为将b赋值给a,并且如果a不等于0,则c=0,当a等于0,则c=1,这与原设计的意图完全不同。应将条件表过式更改为:

a==b

8、语句后面漏加分号

C语言规定语句末尾必须有分号,分号是C语句不可缺少的一部分,

例如:

main ()

{

unsigned int i,sum;

sum=0;

for (i=0;i<10;i++)

{sum+=i}

}

很多初学者认为用大括号括起就不必加分号,这是错误的,即使该语句用大括号括起来,也必须加入分号。在复合语句中,初学者往往容易漏写最后一个分号。上例应改为如下形式:

main ()

{

unsigned int i,sum;

sum=0;

for (i=0;i<10;i++)

{sum+=i;}

}

当漏写分号而出错,光标将停留在漏写分号的下一行。

9、在不该加分号的地方加了分号

#include “io8515v.h”;

由于伪指令不是C程序语句,因此后面不能加分号。

初学者也常在判断语句的条件表达式后面加入分号,

例如:

main ()

{

unsigned int i,sum;

sum=0;

for (i=0;i<10;i++);

sum+=i;

}

在上例中,在for的表达式后面中入分号,则C编译认为循环体是一个空操作,这与设计者的本意不符。


c源程序常见错误分析》有 2 条评论

  1. vcking 说:

       各位大侠:

      哪里有VC++的精简版啊 给一个URL给我 THANK 我的QICQ411342104

  2. blue 说:

    楼上的。看这里。不过精简的不好。。

    http://www.vcgood.com/forum_posts.asp?TID=450&PN=1&T PN=1

留下一个回复