常见错误和程序调试课件.ppt
- 【下载声明】
1. 本站全部试题类文档,若标题没写含答案,则无答案;标题注明含答案的文档,主观题也可能无答案。请谨慎下单,一旦售出,不予退换。
2. 本站全部PPT文档均不含视频和音频,PPT中出现的音频或视频标识(或文字)仅表示流程,实际无音频或视频文件。请谨慎下单,一旦售出,不予退换。
3. 本页资料《常见错误和程序调试课件.ppt》由用户(晟晟文业)主动上传,其收益全归该用户。163文库仅提供信息存储空间,仅对该用户上传内容的表现方式做保护处理,对上传内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知163文库(点击联系客服),我们立即给予删除!
4. 请根据预览情况,自愿下载本文。本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
5. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007及以上版本和PDF阅读器,压缩文件请下载最新的WinRAR软件解压。
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 常见 错误 程序 调试 课件
- 资源描述:
-
1、第第1616章章 常见错误和程序调试常见错误和程序调试16.1常见错误分析常见错误分析16.2程序调试程序调试 要真正学好要真正学好C、用好、用好C并不容易,并不容易,“灵活灵活”固然是固然是好事,但也使人难以掌握,尤其是初学者往往出好事,但也使人难以掌握,尤其是初学者往往出了错还不知怎么回事。了错还不知怎么回事。C编译程序对语法的检查编译程序对语法的检查不如其他高级语言那样严格不如其他高级语言那样严格(这是为了给程序人员这是为了给程序人员留下留下“灵活灵活”的余地的余地)。因此,往往要由程序设计。因此,往往要由程序设计者自己设法保证程序的正确性。调试一个者自己设法保证程序的正确性。调试一个C
2、程序程序要比调试一个要比调试一个PASCAL或或FORTRAN程序更困难程序更困难一些。需要不断积累经验,提高程序设计和调试一些。需要不断积累经验,提高程序设计和调试程序的水平。程序的水平。C语言有些语法规定和其他高级语言不同,学习语言有些语法规定和其他高级语言不同,学习过其他高级语言的读者往往按照使用其他高级语过其他高级语言的读者往往按照使用其他高级语言的习惯来写言的习惯来写C程序,这也是出错的一个原因。程序,这也是出错的一个原因。16.1常见错误分析常见错误分析 下面将初学者在学习和使用下面将初学者在学习和使用C语言语言(不包括不包括C+)时时容易犯的错误列举出来,以起提醒的作用。这些容易
3、犯的错误列举出来,以起提醒的作用。这些内容在以前各章中大多已谈到,为便于查阅,在内容在以前各章中大多已谈到,为便于查阅,在本章中集中列举,供初学者参考,以此为鉴。本章中集中列举,供初学者参考,以此为鉴。(1)忘记定义变量。如:忘记定义变量。如:main()x=3;y=6;printf(%dn,x+y);C要求对程序中用到的每一个变量都必须定义其要求对程序中用到的每一个变量都必须定义其类型,上面程序中没有对类型,上面程序中没有对x、y进行定义。应在函进行定义。应在函数体的开头加数体的开头加intx,y;这是学过这是学过BASIC和和FORTRAN语言的读者写语言的读者写C程序时常见的一个错误。程
4、序时常见的一个错误。在在BASIC语言中,可以不必先定义变量类型就可语言中,可以不必先定义变量类型就可直接使用。在直接使用。在FORTRAN中,未经定义类型的变中,未经定义类型的变量按隐含的量按隐含的I-N规则决定其类型,而规则决定其类型,而C语言则要求语言则要求对用到的每一个变量都要在本函数中定义对用到的每一个变量都要在本函数中定义(除非已除非已定义为外部变量定义为外部变量)。(2)输入输出的数据的类型与所用格式说明符不一输入输出的数据的类型与所用格式说明符不一致。致。例如,若例如,若a已定义为整型,已定义为整型,b已定义为实型。已定义为实型。a=3;b=45;printf(%f%dn,a,
5、b);编译时不给出出错信息,但运行结果将与原意不编译时不给出出错信息,但运行结果将与原意不符,输出为符,输出为 000000016402 它们并不是按照赋值的规则进行转换它们并不是按照赋值的规则进行转换(如把如把45转转换成换成4),而是将数据在存储单元中的形式按格式,而是将数据在存储单元中的形式按格式符的要求组织输出符的要求组织输出(如如b占占4个字节,只把最后两个个字节,只把最后两个字节中的数据按字节中的数据按%d,作为整数输出,作为整数输出)。(3)未注意未注意int型数据的数值范围。型数据的数值范围。一般微型计算机上使用的一般微型计算机上使用的C编译系统,对一个整编译系统,对一个整型数
6、据分配两个字节。因此一个整数的范围为型数据分配两个字节。因此一个整数的范围为-215215-1,即,即-3276832767。常见这样的程序。常见这样的程序段:段:int num;num=89101;printf(%d,num);得到的却是得到的却是23565,原因是,原因是89101已超过已超过32767。两。两个字节容纳不下个字节容纳不下89101,则将高位截去。见图,则将高位截去。见图16.1。即将超过低即将超过低16位的数截去。即将位的数截去。即将89101减去减去216(即即16位二进制所形成的模位二进制所形成的模)。89101-65536=23565。图图16.1 有时还会出现负数
7、。例如有时还会出现负数。例如 num=196607;输出得输出得-1。因为。因为196607的二进制形式为的二进制形式为 00 00 00 00 00 00 00 1011 11 11 11 11 11 11 11 去掉高位去掉高位10,低,低16位的值是位的值是-1(-1的补码是:的补码是:1111111111111111)。对于超过整个范围的数,要用对于超过整个范围的数,要用long型,即改为型,即改为 long intnum;num=89101;printf(%ld,num);请注意,如果只定义请注意,如果只定义num为为long型,而在输出时型,而在输出时仍用仍用“%d”说明符,仍会出
8、现以上错误。说明符,仍会出现以上错误。(4)输入变量时忘记使用地址符。如:输入变量时忘记使用地址符。如:scanf(%d%d,a,b);这是许多初学者刚学习这是许多初学者刚学习C语言时一个常见的疏忽,语言时一个常见的疏忽,或者说是习惯性的错误,因为在其他语言中在输或者说是习惯性的错误,因为在其他语言中在输入时只需写出变量名即可,而入时只需写出变量名即可,而C语言要求指明语言要求指明“向哪个地址标识的单元送值向哪个地址标识的单元送值”。应写成。应写成scanf(%d%d,&a,&b);(5)输入时数据的组织与要求不符。输入时数据的组织与要求不符。用用scanf函数输入数据,应注意如何组织输入函数
9、输入数据,应注意如何组织输入数据。假如有以下数据。假如有以下scanf函数:函数:scanf(%d%d,&a,&b);有人按下面的方法输入数据:有人按下面的方法输入数据:3,4 这是错的。数据间应该用空格来分隔。读者可以这是错的。数据间应该用空格来分隔。读者可以用用 printf(%d%d,a,b);来验证一下。应该用以下方法输入:来验证一下。应该用以下方法输入:34 如果如果scanf函数为函数为 scanf(%d,%d,&a,&b);对对scanf函数中格式字符串中除了格式说明符外,函数中格式字符串中除了格式说明符外,对其他字符必须按原样输入。因此,应按以下方对其他字符必须按原样输入。因此
10、,应按以下方法输入:法输入:3,4 此时如果用此时如果用“34”反而错了。还应注意,不能企图反而错了。还应注意,不能企图用用 scanf(input a&b:%d,%d,&a,&b);想在屏幕上显示一行信息:想在屏幕上显示一行信息:input a&b:然后在其后输入然后在其后输入a和和b的值,这是不行的。这是由的值,这是不行的。这是由于有的读者以为于有的读者以为scanf具有具有BASIC语言中的语言中的INPUT语句的功能语句的功能(先输出一个字符串,再输入变量的值先输出一个字符串,再输入变量的值)。如果想在屏幕上得到所需的提示信息,可以另加如果想在屏幕上得到所需的提示信息,可以另加一个一个
11、printf函数语句:函数语句:printf(input a&b:);scanf(%d,%d,&a,&b);(6)误把误把“=”作为作为“等于等于”比较符。比较符。在许多高级语言中,用在许多高级语言中,用“=”符号作为关系运算符符号作为关系运算符“等于等于”。例如,在。例如,在BASIC或或PASCAL程序中都程序中都可以写可以写 if(a=b)then 但在但在C语言中,语言中,“=”是赋值运算符,是赋值运算符,“=”才是关才是关系运算符系运算符“等于等于”。如果写成。如果写成 if(a=b)printf(a equal to b);C编译系统将编译系统将(a=b)作为赋值表达式处理,将作为
12、赋值表达式处理,将b的值的值赋给赋给a,然后判断,然后判断a的值是否零,若为非零,则作的值是否零,若为非零,则作为为“真真”;若为零作为假。如果若为零作为假。如果a的值为的值为3,b的值的值为为4,ab,按原意不应输出,按原意不应输出“ae q u a lt ob”。而现在。而现在先将先将b的值赋给的值赋给a,a也为也为4,赋值表达式的值为,赋值表达式的值为4。if语句中的表达式值为真语句中的表达式值为真(非零非零),因此输出,因此输出“ae q u a lt o b”。这种错误在编译时是检查不出来的,但运行结果这种错误在编译时是检查不出来的,但运行结果往往是错的。而且由于习惯的影响,程序设计
13、者往往是错的。而且由于习惯的影响,程序设计者自己往往也不易发觉。自己往往也不易发觉。(7)语句后面漏分号。语句后面漏分号。C语言规定语句末尾必须有分号。分号是语言规定语句末尾必须有分号。分号是C语句不语句不可缺少的一部分。这也是和其他语言不同的。有可缺少的一部分。这也是和其他语言不同的。有的初学者往往忘记写这一分号。如:的初学者往往忘记写这一分号。如:a=3 b=4 编译时,编译程序在编译时,编译程序在“a=3”后面未发现分号,就后面未发现分号,就把下一行把下一行“b=4”也作为上一行的语句的一部分,也作为上一行的语句的一部分,这就出现语法错误。有时编译时指出某行有错,这就出现语法错误。有时编
14、译时指出某行有错,但在该行上并未发现错误,应该检查上一行是否但在该行上并未发现错误,应该检查上一行是否漏了分号。漏了分号。如果用复合语句,有的学过如果用复合语句,有的学过PASCAL语言的读者语言的读者往往漏写最后一个语句的分号,如:往往漏写最后一个语句的分号,如:t=a;a=b;b=t 在在PASCAL中分号是两个语句间的分隔符而不是语中分号是两个语句间的分隔符而不是语句的一部分,而在句的一部分,而在C中,没有分号的就不是语句。中,没有分号的就不是语句。(8)在不该加分号的地方加了分号。在不该加分号的地方加了分号。例如:例如:if(ab);printf(aislarger than bn);
15、本意为当本意为当ab时输出时输出“aislarger than b”的信的信息。息。但由于在但由于在if(ab)后加了分号,因此后加了分号,因此if语句到此结束。语句到此结束。即当即当(ab)为真时,执行一个空语句。本来想为真时,执行一个空语句。本来想ab时不输出上述信息,但现在时不输出上述信息,但现在printf函数语句并不从函数语句并不从属于属于if语句,而是与语句,而是与if语句平行的语句。不论语句平行的语句。不论ab还是还是ab,都输出,都输出“a is larger than b”。又如:又如:for(i=0;i10;i+);scanf(%d,&x);printf(%dn,x*x);
16、本意为先后输入本意为先后输入10个数,每输入一个数后输出它个数,每输入一个数后输出它的平方值。由于在的平方值。由于在for()后加了一个分号,使循后加了一个分号,使循环体变成了空语句。只能输入一个整数并输出它环体变成了空语句。只能输入一个整数并输出它的平方值。的平方值。总之,在总之,在if、for、while语句中,不要画蛇添足多语句中,不要画蛇添足多加分号。加分号。(9)对应该有花括弧的复合语句,忘记加花括弧。对应该有花括弧的复合语句,忘记加花括弧。如:如:sum=0;i=1;while(i=100)sum=sum+i;i+;本意是实现本意是实现1+2+100,即,即i。但上面的语句只。但上
17、面的语句只是重复了是重复了sum+1的操作,而且循环永不终止。因为的操作,而且循环永不终止。因为i的值始终没有改变。错误在于没有写成复合语句的值始终没有改变。错误在于没有写成复合语句形式。因此形式。因此while语句的范围到其后第一个分号为语句的范围到其后第一个分号为止。语句止。语句“i+;”不属于循环体范围之内。应改不属于循环体范围之内。应改100i=0为为while(i=100)sum=sum+i;i+;(10)括弧不配对。括弧不配对。当一个语句中使用多层括弧时常出现这类错当一个语句中使用多层括弧时常出现这类错误,纯属粗心所致。如:误,纯属粗心所致。如:while(c=getchar()!
18、=#)putchar(c);少了一个右括弧。少了一个右括弧。(11)在用标识符时,忘记了大写字母和小写字母在用标识符时,忘记了大写字母和小写字母的区别。的区别。例如:例如:main()inta,b,c;a=2;b=3;C=A+B;printf(%d+%d=%,A,B,C);编译时出错。编译程序把编译时出错。编译程序把a和和A认作是两个不同的认作是两个不同的变量名处理,同样变量名处理,同样b和和B,c和和C都分别代表两个不都分别代表两个不同的变量。同的变量。(12)引用数组元素时误用了圆括弧。引用数组元素时误用了圆括弧。如:如:main()int i,a(10);for(i=0;i10;i+)s
19、canf(%d,&a(i);C语言中对数组的定义或引用数组元素时必须用语言中对数组的定义或引用数组元素时必须用方括弧。方括弧。(13)在定义数组时,将定义的在定义数组时,将定义的“元素个数元素个数”误认误认为是为是“可使用的最大下标值可使用的最大下标值”。main()int a10=1,2,3,4,5,6,7,8,9,10;int i;for(i=1;i=10;i+)printf(%d,ai);想输出想输出a1到到a10。但一些初学者常犯的错误。但一些初学者常犯的错误。C语言规定定义时用语言规定定义时用a10,表示,表示a数组有数组有10个元素,个元素,而不是可以用的最大下标值为而不是可以用的
20、最大下标值为10。数组只包括。数组只包括a0到到a910个元素,因此用个元素,因此用a10就超出就超出a数组的范围数组的范围了。了。(14)对二维或多维数组的定义和引用的方法不对。对二维或多维数组的定义和引用的方法不对。main()int a5,4;printf(%d,a1+2,2+2);对二维数组和多维数组在定义和引用时必须将每对二维数组和多维数组在定义和引用时必须将每一维的数据分别用方括弧括起来。上面一维的数据分别用方括弧括起来。上面a5,4应应改为改为a54,a1+2,2+2应改为应改为a1+22+2。根。根据据C的语法规则,在一个方括弧中的是一个维的的语法规则,在一个方括弧中的是一个维
21、的下标表达式,下标表达式,a1+2,2+2中方括弧中的中方括弧中的“1+2,2+2”是一个逗号表达式,它的值是第二个数值表达式是一个逗号表达式,它的值是第二个数值表达式的值,即的值,即2+2的值为的值为4。所以。所以a1+2,2+2相当于相当于a4。而。而a4是是a数组的第数组的第4行的首地址。因此执行行的首地址。因此执行printf函数输出的结果并不是函数输出的结果并不是a34的值,而是的值,而是a数组第数组第4行的首地址。行的首地址。(15)误以为数组名代表数组中全部元素。误以为数组名代表数组中全部元素。例如:例如:main()int a4=1,3,5,7;printf(%d%d%d%dn
22、,a);企图用数组名代表全部元素。在企图用数组名代表全部元素。在C语言中,数组语言中,数组名代表数组首地址,不能通过数组名输出名代表数组首地址,不能通过数组名输出4个整数。个整数。(16)混淆字符数组与字符指针的区别。混淆字符数组与字符指针的区别。main()char str4;str=Computer and c;printf(%sn,str);编译出错。编译出错。str是数组名,代表数组首地址。在编是数组名,代表数组首地址。在编译时对译时对str数组分配了一段内存单元,因此在程序数组分配了一段内存单元,因此在程序运行期间运行期间str是一个常量,不能再被赋值。因此,是一个常量,不能再被赋值
23、。因此,str=“Computer and c”是错误的。如果把是错误的。如果把“char str4;”改成改成“charstr;”,则程序正确。此时,则程序正确。此时str是指向字符数据的指针变量,是指向字符数据的指针变量,str=“Computer and c”是合法的,它将字符串的首地址赋给指针变量是合法的,它将字符串的首地址赋给指针变量str,然后在,然后在printf函数语句中输出字符串函数语句中输出字符串“Computer and c”。因此应当弄清楚字符数组与字符指针变量用法的因此应当弄清楚字符数组与字符指针变量用法的区别。区别。(17)在引用指针变量之前没有对它赋予确定的值。在
24、引用指针变量之前没有对它赋予确定的值。main()char*p;scanf(%s,p);没有给指针变量没有给指针变量p赋值就引用它,编译时给出警告赋值就引用它,编译时给出警告信息。应当改为信息。应当改为charp,c20;p=c;scanf(%s,p);即先根据需要定义一个大小合适的字符数组即先根据需要定义一个大小合适的字符数组c,然,然后将后将c数组的首地址赋给指针变量数组的首地址赋给指针变量p,此时,此时p有确定有确定的值,指向数组的值,指向数组c。再执行。再执行scanf函数就没有问题了,函数就没有问题了,把从键盘输入的字符串存放到字符数组把从键盘输入的字符串存放到字符数组c中。中。(1
25、8)switch语句的各分支中漏写语句的各分支中漏写break语句。语句。例如:例如:switch(score)case 5:printf(ery good!);case 4:printf(Good!);case 3:printf(Pass!);case 2:printf(Fail!);defult:printf(data error!);上述上述switch语句的作用是希望根据语句的作用是希望根据score(成绩成绩)打印打印出评语。但当出评语。但当score的值为的值为5时,输出为时,输出为 ery Good!Good!Pass!Fail!data error!原因是漏写了原因是漏写了br
展开阅读全文