1.程序开发过程中出现的错误类型
我们在进行程序设计时,不可避免地会犯错误。程序中的错误可以分为三类:编译错误、运行时错误和逻辑错误。
1.1编译错误
编译错误(Compile errors)又称为编译时错误(Compiling-time errors): 是由于错误的编码产生的。例如关键字拼写错误、将中文标点符号当成英文符号使用、遗漏了某些必要的标点符号或者使用了一个没有定义的标识符。
编译错误一般都是语法错误,当编译器对程序进行语法检查时,都能发现这些错误,并能够指出产生错误的位置(标出行号)。我们可以根据编译出错信息指出的行号到对应的源代码行改正错误,重新编译源程序。只有当所有的编译错误被改正后,才能通过编译检查,产生目标代码文件。
改正编译错误的关键是要能正确理解编译器给出的编译错误信息。VC++环境中的编译、链接错误信息是用英文表示的。对于英文基础薄弱的读者,可以参考本实验指导书后面的“VC++
译、链接常见错误和警告信息中英文对照”。
通常情况下,一个语法错误可能产生多条编译错误信息,这是由于株连错误造成的,建议读者在处理编译错误时,到第一个出现错误的位置改正后重新编译。这样能够避免被株连错误迷惑。值得指出的是,现在大部分编译器对错误的定位不精确,如果在编译器指出的行没有发现错误,应该向前查错误。例如,当提示第10行发生错误时,如果在第10行没有发现错误,请从第10行开始往前查错误并修改之。
1.2运行时错误
运行时错误(Run-time errors)是在程序的运行阶段出现的,当运行环境检测到程序的某些操作无法执行,例如除数为零时,就会出现运行时错误。当运行环境检测到程序的某些操作是被禁止的,也会产生运行时错误。例如,访问数组时超越数组的边界,空指针引用(NULL pointer assignment,空指针赋值,即有指针未赋具体地址就使用了)等等。
1.3逻辑错误
逻辑错误(Logic errors):当程序没有按照程序员的意图执行时,就表明程序中存在逻辑错
误。一个应用程序可能既没有语法错误,运行时也没有执行任何无效的操作,但是有可能产生错误的结果,这种错误结果一般都是程序内部的逻辑错误造成的。只有通过测试应用程序并分析它产生的结果,我们才能核实应用程序是否正确地执行了。
当然,如果输入了错误的或者无效的数据,执行程序后肯定也得不到正确的结果。软件行业中有一句名言“输入的是垃圾,输出也是垃圾。(Garbage in, garbage out.)”因此,一般的实用程序还需要对输入数据的正确性和有效性进行检验。
很显然,必须出并改正程序中的错误,才能得到正确的执行结果。对于语法错误,我们可以根据编译错误信息指出的位置和错误原因来改正错误。请记住:编译器不能发现程序中的逻辑错误和运行时错误。我们可以通过仔细阅读源程序来发现逻辑错误,还可以借助开发环境中提供的调试工具来查程序中的逻辑错误和运行时错误。
所谓调试(debug)就是定位程序中的错误并改正错误的过程。为了帮助程序员出并改正程序中的错误,微软公司在Visual C++6.0环境中集成了调试器(debugger, 调试器就是调试工具(debugging tools)。下面我们将介绍Visual C++6.0中的调试工具及其使用方法。
2. C语言容易错误的地方
2.1基础知识和数据类型、表达式
1.拼写错误,尤其是include,main,void,float等词。
2.{},[],(),‘’,“”不配对。解决这个问题最好的方法就是每当写这些符号的时候就先写成一对,然后再在中间加内容。
3.忘记在语句的末尾加分号,或在预处理命令后多加分号。记住:每一个语句的后边都要加分号,而预处理命令并不是语句,所以不加分号,他们必须每行一条,不能把多个命令写在一行。
4.混淆/\;注释对应的符号是/*  */,而转义字符是以\开头,除号是/
5.printf()scanf()的参数设置有误,主要表现在以下几方面:
编译器错误类型不匹配的问题。(例如:有float a=3.5,但输出的时候printf(“a=%d”,a);则屏幕上会显示出a=0.00000或者提示其它运行错误)。基本原则是:float对应%f, int对应%d, char对应%c
个数不匹配。无论是哪个函数,都可以有n个参数,第一个永远是“”括起来的内容,表示输出
格式。剩下的n-1个是输出的变量或者输入的变量的地址。需要注意的是,如果后边有n-1个参数,那么前边一定对应n-1%f一类的格式说明符。
scanf()中变量前忘了加&。记住:scanf()中变量前要有&(但后边学到的字符数组名和指针前不用加)
6.定义标识符的时候经常出现使用非法字符的情况,例如:标识符中不能用空格,也就是说不能有这样的定义:int radium of circle;一般情况下可用下划线将三个单词连接在一起。
7.在使用变量前未定义,或未初始化。例如:若下边的sum未定义,则在编译时会提示相应的错误信息,而若未初始化为0,则求和的结果一定是错误的。
 void main()
{ int I,a[10], sum=0;  /*只要下边要用,这个定义就必须要有,一般情况下也要有初始值*/
for(I=0;I<10;I++) sum+=a[I];
printf(“%d”,sum);
}
8.符号常量定义错误。例如:#define PI=3.14159,这里的=应该换成空格。
9.计算错误。主要注意:++,――和其它运算符一起运算时,除根据优先级进行计算时,还要考虑先后位置的特殊含义;数据类型不一致时发生的自动转换也会导致计算的误差;还要注意求模结果的符号与被除数相同;某些特殊情况下 使用懒惰求值法。
10.不能除以0,要做合法性检查;
11.类型溢出。记住每种数据类型的取值范围,确保数据在所定义类型范围之内;
12.数学表达式的格式有误。常见的有:(1)数学与C语言运算表达式的混淆(例如:=表示赋值,而= =才表示我们数学中的相等关系)。(2)、忽略了运算的优先级。解决这个问题的最好方法就是写数学表达式时不要从左到右,而是按优先级的顺序写,写完优先级高的一个表达式后加上()再写下一级的表达式,例如:计算梯形的面积时,要s=((a+b)*h)/2,不要1/2*a+b*h. (3)忽略了计算和赋值时的自动转换。例如:float half=1/2;这样,因为=右边是整数相除的结果为整数0,不会得到0.5存入half,进而会影响下边的计算结果。要想不在
这儿绊跟头,当计算不同类型的数据时,一定注意会不会出现引起错误的自动转换,建议最好加上强制转换。(4)赋值号左边不是变量,例如:若有#define PI 3.14,程序中又出现PI=3.14159。又例如:f(n)=f(n-1)*n(这是典型的数学语言,在C语言中右边的乘积不能正确存储,而左边又是一个函数调用)。
2.2流程控制
1.丢掉语句结束标记“;”,尤其是for语句中表达式后或do-while语句后的分号,或在预处理命令后边、while()后、for()后加“;”;
2.If语句或循环语句中逻辑表达式或关系表达式书写错误。一定要注意C语言的条件与数学表达式的区别(例如我们数学中经常写到的0x9,C语言中应该写成x>=0&&x<=9)。
3.if-else嵌套时不配对。最好在写每个条件时要用两个{}分别将两个分支先括起来,再添加其中的语句,以保证其配对不易错。
4.switch()语句中的格式不正确。()中的表达式结果一定是一些明确的值,不能是区间;表达式的所有可能结果要列在case后边,case与常量之间有一空格,不要丢掉必要的break;
5.随意修改循环控制变量i的值,导致循环次数的改变,尤其是当循环有嵌套时。在循环体中,不要将循环控制变量进行另外的改变。
6.分不清什么情况下用双重循环,什么情况下用两个控制变量写成一重循环。当I不变,j又循环一遍的时候用双重循环。当I,j同时变化的时候用一重循环,此时,循环控制变量有两个,但条件只写一个就可以,因为另一个总是进行相应的变化的。
7.忽略循环体与循环控制变量的关系。其实,很多情况下,循环控制变量都在循环体中起到非常重要的作用。应该利用上这种关系。
2.3数组与指针
1.字符串的输入有错误:主要表现在使用scanf()gets()时加了&,或输入字符串时用循环,(这样的话,字符个数无论多长,都不会为自动加\0,将来引用的时候也就不能以字符串的形式引用。)
2.对字符串的处理中,循环条件仍然写成I<N。由于字符串是不定长的,所以循环条件一般为str[I]!=’\0’ I<strlen(str)
3.而输入所对应的变量是指针时(常见的有:输入的变量是字符数组名或指向字符串的指针)不能加&。
4.指针定义后未赋值就引用。如果在定义时不知道赋什么值,可以用p=NULL赋初值,以避免引起的灾难性错误。
5.分不清p和*p。前者是指针,即地址,后者表示指针所间接引用的数据,但如果是二级指针或多级指针,取*以后得到的仍然可能是地址。
2.4函数
1.函数定义的时候,函数头部加分号,而函数声明的地方忘了加分号
2.函数实参格式不对,主要表现在:给出实参时,多给出数组类型,或者,形参是数组int a[];的时候,给出的a[]a[I].
3.递归时忘了设置边界条件,这样易造成死循环调用。
4.使用函数之前未声明(包括C库函数的声明)。建议大家,将所定义的一切函数都在程序
开始的预处理命令后加上函数原型的声明,这样做不仅可以避免错误,而且整个程序的结构看起来更清楚。
3.编译错误
(1)error C2001: newline in constant
编号:C2001
直译:在常量中出现了换行。
错误分析:
1)字符串常量、字符常量中是否有换行。
2)在这句语句中,某个字符串常量的尾部是否漏掉了双引号。
3)在这语句中,某个字符创常量中是否出现了双引号字符“"”,但是没有使用转义符“\"”。
4)在这句语句中,某个字符常量的尾部是否漏掉了单引号。
5)是否在某句语句的尾部,或语句的中间误输入了一个单引号或双引号。
(2)error C2015: too many characters in constant
编号:C2015
直译:字符常量中的字符太多了。
错误分析:
单引号表示字符型常量。一般的,单引号中必须有且只能有一个字符(使用转义符时,转义符所表示的字符当作一个字符看待),如果单引号中的字符数多于4个,就会引发这个错误。
另外,如果语句中某个字符常量缺少右边的单引号,也会引发这个错误,例如:
if (x == 'x || x == 'y') { … }
值得注意的是,如果单引号中的字符数是2-4个,编译不报错,输出结果是这几个字母的ASC码作为一个整数(int,4B)整体看待的数字。
(3)error C2137: empty character constant