计算机二级C语言程序设计上机改错题技巧总结.docx
计算机二级C语言程序设计上机改错题技巧总结一、 改错题之方法步骤1. 上机考试中改错题占30分,一般有两个地方有错误,题型简单2. 做改错题时先看出错的地方,分析语法错误,如果能用C语言的语法判断出错误,改之即可3. 没有语法错误即分析逻辑错误,逻辑错误可以从几个方面分析:(1) 从题目的要求中找到错误,例如:题目要求计算s=1+1/2+1/3+1/n,那么循环的范围就应该是for(i=0;i<=n;i+),但是考试中经常将其写为:for(i=0;i<n;i+)(2) 根据题目中的关键字改错,例如:题目中要求从小到大排序,则“从小到大”就是关键字(3) 重点注意函数的调用、函数的返回值类型,函数的形参,这个是上机考试中的重点(4) 注意细节,请参考以下为考生总结的知识(5) 多练习,多思考,多总结二、 改错题之语法错误1. 关键字出错(1) 关键字出错,指的是C语言中的关键字出错。C语言的关键字都是小写字母,并且在VC+6.0中以蓝色字体显示,如果常见的关键字是黑色字体,可以断定关键字出错;(2) if关键字出错1) 举例:上机题库P79第6题/*found*/ If(substrk+1='0')2) 类似题目:12、35题(3) double关键字出错1) 举例:上机题库P86第23题/*found*/double fun(double a, dounle x0)2) 类似题目:36、51(4) while关键字出错1) 举例:上机题库P77第2题/*found*/ wihle( q>p )2) 类似题目:(5) return关键字出错1) 举例:上机题库P101第54题/*found*/ Return(Value);2) 类似题目:862. 格式出错(1) C语言中的语句、基本结构、函数等都有相应的规范格式,程序中将这些格式书写错误,将导致编译器无法识别程序,因此导致出错;(2) 语句末尾少分号1) 分号是语句结束的标志,每个语句都必须以分号结尾2) 举例:上机题库P77第2题第一个出错点/*found*/ q=p+i3) 类似的题目:28题第一空、29题第1空、34题第2空、37题第2空、48题第2空,49第1空,59第2空、70、79、97(3) for结构格式出错1) for结构的格式:for(表达式1;表达式2;表达式3) 循环体语句 ,for结构体中表达式必须以分号隔开,改错题中经常出现将for结构中分号错写为逗号的情况;2) 举例:上机题库P77第3题第2个出错点/*found*/ for( i = 100, i <= m, i += 100 )3) 类似的题目:6题第一空、14题第二空、53第1空、73、86第1空(4) if结构格式出错1) if结构的格式:if(表达式),if后面必须加上小括号2) 举例:P93第37第1个出错点/*found*/ if n = = 03) 类似题目:59题第1空(5) 数组格式出错1) 一维数组的定义形式:类型名 数组名常量表达式,谨记是数组的独有特征2) 举例:P89第28题第2个出错点/*found*/ t(k)=b(j);3) 举例2:P97第1个出错点(二维数组不能省略列下标)/*found*/int fun(int n, int xx)4) 类似题目:95题第1空(6) 函数格式出错1) 记住函数名后面一定是花括号,不能是方括号,方括号只有在数组下标中用到2) 举例:P86第22题第1出错点/*found*/ n=strlen aa ;3) scanf(输入控制,输入数据列表),scanf函数中输入数据列表要求为变量的地址4) 举例2:P95第41题第2出错点/*found*/ scanf("%d",aij);5) 类似题目:第80题第1空(7) 命令行格式错误1) #include “stdio.h”,include和define前面都必须有#2) 举例:P84第17题第1出错点/*found*/include (stdio.a)3. 使用了未定义的变量:变量必须先定义后执行(1) 0(零)和o(字母欧)的区别1) 区别:零像鸡蛋,字母欧像乒乓球2) 举例:上机题库P85第19题第一个出错点/*found*/ a2=k/1o;3) 类似题目:19题第一空、23题第二空、43题第二空、51题第二空、77、98、(2) P(大)和p(小)的区别1) 区别:一般在定义时候p为小写,但是在使用的时候变为大写P,做题时候一定要细心2) 举例:上机题库P91第33题第一出错点fun(int *a,int n,int x) int p=0,i; an=x; while( x!=ap )p=p+1;/*found*/ if(P=n) return -1;3) 类似题目:33题第一空、82(3) 普通变量大小写的区别1) 定义时候和使用时候变量的大小写不一致,违背了变量必须“先定义后执行”的原则2) 举例:18题第2空int fun ( float *s, int n, float *aver ) float ave, t = 0.0 ; int count = 0, k, i ; for ( k = 0 ; k < n ; k+ )/*found*/ t = sk ; ave = t / n ; for ( i = 0 ; i < n ; i+ ) if ( s i < ave ) count+ ;/*found*/ *aver = Ave ;/*分析:定义时候float ave, t = 0.0 ;ave为小写,但是此时使用时候却是Ave大写,因此出错*/ return count ;3) 类似题目:18题第2空、22题第2空、38题第1空、54题第2空三、 改错题之逻辑错误1. 运算符号出错:(1) C语言中除号是“/”不是“”(2) 赋值号(=)和等号(= =)混淆1) 记住在if语句中出现赋值号(=)一般是讲它改为等号(= =)。2) 类似题目:14题第一空和29题第二空都是这样的错误、87、91、942. 表达式的取值范围出错:(1) 当遇到计算表达式的累加或是累减时,一定用到循环来完成相应的操作。(2) 例如:改错题第七题中要求计算t=1-1/2*2-1/3*3-.-1/m*m这个表达式,在这个题目中第一空有这样的for语句:for(i=2;i<m;i+)y-=1/(i*i);从题目中不难发现i是一个变量,它的取值范围从整数2开始到m-1结束,最终计算的结果为 y=1-1/2*2-1/3*3-.-1/(m-1)*(m-1)和题目要求的不符,因此将i<m改为i<=m。(3) 做这样的题目看两点:一看题目中的表达式,二看for语句中变量的取值范围。(4) 类似题目:31题第二空、52第2空、81、88、3. C语言中的除法运算:(1) 这个考点同样考查表达式的相关知识。(2) 例如:改错题第七题中要求计算t=1-1/2*2-1/3*3-.-1/m*m这个表达式,在这个题目中第一空有这样的for语句:for(i=2;i<m;i+)y-=1/(i*i);刚才我们将了表达式的范围,现在我们考虑y-=1/(i*i);是否正确?赋值号左边问题不大,我们看看右边1/(i*i),其中i是整数,那么i*i也是整数,在C语言中1除以比它大的数时值为零。这样和题目的要求就不相符。(3) 记住:1/2的值为零,只有1.0/2或是1.0/2.0的值为0.5。(4) 类似题目:39题第一空也同样是这种错误、57、764. 找最大值和排序问题:(1) 遇到这种题目在改错题中最容易考查的是大于(>)、小于(<)等符号的变化,当你查找出错的地方没有语法错误,也没有使用了未定义的变量时,记住这个口诀:将大于改为小于,将小于改为大于!(2) 类似题目:11题第二空、12题第一空和13题第二空、4题第二空、585. 函数的返回值及其参数:重点int fun(int x,int y) main() if(x>y)return x; int a,b,c;else return y; scanf(“%d%d”,&a,&b); c=fun(a,b); printf(“最大值为:%d”,c);(1) 函数的返回类型:1) 在上面的例子中fun函数的返回值是int,如果将int改为double,很明显是错误的。从两个地方来看,首先return x;中x的类型必须和fun函数中的返回值相同,x是整型。其次c=fun(a,b);中fun函数将一个返回值赋值给了变量C,说明函数返回的值一定是整型。2) 记住:函数调用、函数的返回值和ruturn中数据类型必须一致!3) 类似题目:1.1、3.1、16.1、19.2、34.1、36.1、40.1、48.1、71都死这样的错误,(注1.1表示第一题第一空)。78、89、90、(2) return后面的表达式及其类型:19.2、30.2、39.2(3) 函数传递的参数:int fun(int *x,int *y) main() if(*x>*y)return *x; int a,b,c;else return *y; scanf(“%d%d”,&a,&b); c=fun(&a,&b); printf(“最大值为:%d”,c);1) 一定注意:传递的是值还是地址。比较上面两个例子大家很容易区别传递的是值还是地址,第一个例子调用函数的时候传递的是值,因此在fun函数的形参中x和y仅仅是普通的变量,而第二个例子中调用函数的时候传递的是地址,因此在fun函数的形参中x和y就是两个指针变量。2) 类似题目:5.1、13.1、17.2、20.1、25.2、52.1、56、84、85、100(4) 指针的使用:int fun(int *x,int *y) if(*x>*y)return *x; else return *y; 1) 在此例中fun函数的形参是x和y,它们的类型是整型的指针,那么在fun函数的函数体中使用到x和y的值时一定要区分:x>y比较的是x和y所存变量地址的大小,*x>*y才是比较值的大小。2) 如果存在*x=x+y;的类似情况一定是错误的,必须要将赋值号的左右两边的类型变为一致。 3) 类似题目:5.2、5.3、20.2、24.2、35.1、44.1、72、75、一定特别注意!6. 数组: (1) 数组下标的初始值:数组下标从零开始,到长度减一结束。1) 考试技巧:当出错行在定义部分时,先检查该定义语句有没有语法错误,再看定义语句中的变量有没有赋初始值,最后看变量赋的初始值有没有正确。 2) 定义初始值有窍门,遇到累加时,用来存放累加和的变量一般赋值为零,遇到求阶乘时,用来存放阶乘值的变量一般赋值为一。3) 8.1中的变量k赋值为1,但是数组的下标只能从0开始,因此将1改为0就行了。 (2) 普通变量的初始值:1) 依照上面的技巧,我们同样可以将它延伸到普通变量中。41.1中需要将变量的值进行初始化,题目中有这样的要求:“输出主对角线元素之和”,既然这样我们很肯定初始值应该为零,接着我们看看是给哪个变量赋值?看printf(“sum=%d”,sum);输出的是sum的值,说明sum就是我们要补充在横线上的变量。2) 10.1也可以用同样的方法来分析。 (3) 数组和字符串长度减一:1) 当使用数组中的元素时,最大值只能到数组长度减一。字符串以0作为字符串的结束标志,因此取值最大也只能到字符串长度减一。25.2中输出printf(“%d”,aai);分析程序知道i在此时就是数组的长度,我们要输出数组中的元素,最大就只能到aai-1,因此改为 printf(“%d”,aai-1);就行了。2) 4.1和38.2中考查的是字符串长度减一的问题。(4) 数组下标往后走:1) 当题目要求将一个数组中的元素直接赋值或是以某种方式复制到另一个数组中时,接受数据的数组中的下标要自增。9.1中sj=si;将si中的值复制给sj,联系上下函数段发现i是一个变量被for语句控制,从0开始直到遇到0结束。i一直在加但是j没有变化,因此改为sj+=si;即可。2) 类似的题目有:25.2、26.2、32.1、50.2。7. 特例:(1) 部分题目的解答很特殊,需要根据上下文中的内容来解答题目:例如10.1,在此题目中,定义了变量t=0;根据下面程序中的循环,s=0.0,首先加的是变量t的值,即s的值是从0.0开始的,s得到的值是:s=0.0+0.0+1.0/3+,观察表达式的值,这个少加了第一项1.0的值,所以变量t的初值不能是0.0,只能是1.0,或者是1;(2) 21.2中的这个题目中,表达式的值是从2/1+3/2+5/3即表达式的规律是从第一项开始,以后的每一项的分子是前一项的分子和分母的和,分母为前一项的分子。所以该题目的第二个空中的内容应该是:c=a;a+=b;b+=c;这一条语句中 a代表的变量的分子,b代表的是变量的分母,执行第一个表达式的值,把a的值赋值给变量c,第二个表达式是把分子和分母的和相加复制给变量a,第三个表达式是把原来a的值和变量b的值相加之后得到的值赋值给变量b,则变量b得到的值也为分子分母的和,不满足题意,而分母应该为原来分子的值。第三表达式的值应该改为:b=c;(3) 类似的题目有:33.2、40.2、43.1、47.1、50.18. 素数 11.1 、29、83(1) 判断一个数是否是素数的方法很多,的是只用掌握一种即可,首先掌握判断一个素数的方法;(2) 素数的定义:1) 是自然数 2) 只能被1和本身整除,2到本身减一之间的数不能整除该数。(3) 判断m是否为素数的格式: for(i=2;i<m;i+) if(m%i= =0) break; if(m= =i) 表明m是素数;(4) 素数只能被1和m,那么用2到m-1之间的数除m,如果有一个数能整除m,说明m不是一个素数,就不用再判断后面的数值了,用一条break语句提前结束函数;如果一直都不能整除,则说明m是一个素数。那么用循环语句表示2到m-1之间的数的语句为:for(i=2;i<m;i+),如果在这个范围内的数值有一个数值能整除m,说明m不是一个素数,则可以提前结束循环,break;那么结束之后这个循环中的变量I 的值小于m,如果m是一个素数的时候,则一直都不执行break语句;则循环语句结束的时候是在i=m的时候退出循环。结束循环以后,进行i和m的比较,如果i小于m,则表明m不是一个素数;如果i等于m,则表明m是一个素数。(5) 求素数的题目只要把格式给带进去就可以了。11.1这个题目中只用改为“=”。29题也是一样的解答方法。9. 链表 15、27(1) 链表的题目分为带头结点的和不带头结点的链表;(2) 带头结点的链表:带头结点的链表的第一个空的改为:p=h->next;第二个空改为:p=p->next;.(3) 不带头结点的链表:不带头结点的链表的第一个空的改为:p=h;第二个空改为:p=p->next;.第1题 【程序改错】功能:先将在字符串s中的字符按逆序存放到t串中,然后把s中的 字符按正序连接到t串的后面。例如:当s中的字符串为:“ABCDE”时,则t中的字符串应为: “EDCBAABCDE”。-*/#include <conio.h>#include <stdio.h>#include <string.h> void fun (char *s, char *t) int i; sl = strlen(s); for (i=0; i<sl; i+) ti = ssl-i; for (i=0; i<sl; i+) tsl+i = si; t2*sl = "0"main() char s100, t100; printf("nPlease enter string s:"); scanf("%s", s); fun(s, t); printf("The result is: %sn", t);答案: 1). int i,sl; 2). ti = ssl-i-1; 3). t2*sl = '0' 或 t2*sl = 0;第2题 【程序改错】功能:求出以下分数序列的前n项之和。和值通过函数值返回main 函数。 2/1+3/2+5/3+8/5+13/8+21/13 例如:若n = 5,则应输出:8.391667。-*/#include <conio.h>#include <stdio.h> fun ( int n ) int a, b, c, k; double s; s = 0.0; a = 2; b = 1; for ( k = 1; k <= n; k+ ) s = (double)a / b; c = a; a = a + b; b = c; return c;main( ) int n = 5; printf( "nThe value of function is: %lfn", fun ( n ) );答案: 1). double fun(int n) 2). s = s + (double)a / b; 或 s += (double)a / b; 或 s += a /(double)b; 或 s=s+a/(double)b; 3). return s;第3题 【程序改错】功能:读入一个整数m( 5m20 ),函数getarr调用函数rnd获 得m个随机整数,函数sortpb将这m个随机整数从小到大排序。例如:若输入整数7,则应输出:3 10 17 28 32 36 47。-*/#include "conio.h"#include <stdio.h>sortpb ( int n, int *a ) int i, j, p, t for ( j = 0; j < n-1 ; j+ ) p = j; for ( i = j + 1; i < n ; i + ) if ( ap > aj ) p = i; if ( p = j ) t = aj; aj = ap; ap = t; double rnd ( ) static t = 29, c = 217, m = 1024, r = 0; r =( r*t + c )%m; return( ( double )r/m );getarr( int n, int *x ) int i; for( i = 1; i <= n; i+, x+ ) *x = ( int )( 50*rnd() );putarr( int n, int *z ) int i; for( i = 1; i <= n; i+, z+ ) printf( "%4d", *z ); if ( !( i%10 ) ) printf( "n" ); printf("n");main() int aa20, n; printf( "nPlease enter an integer number between 5 and 20: " ); scanf( "%d", &n ); getarr( n, aa ); printf( "nnBefore sorting %d numbers:n", n ); putarr( n, aa ); sortpb( n, aa ); printf( "nAfter sorting %d numbers:n", n ); putarr( n, aa );答案: 1). int i, j, p, t; 2). if ( ap > ai ) p = i; 3). if ( p != j )第4题 【程序改错】-功能:以下程序能求出1*1+2*2+.+n*n<=1000中满足条件的 最大的n。-*/#include <stdio.h>#include "string.h"main() int n,s; s=n=0; while(s>1000) +n; s+=n*n; printf("n=%dn",&n-1);答案: 1). s=n=0; 2). while(s<=1000) 3). printf("n=%dn",n-1);第5题 【程序改错】-功能:求出a所指数组中最大数和次最大数(规定最大数和次最大 数不在a0和a1中),依次和a0、a1中的数对调。例如:数组中原有的数:7、10、12、0、3、6、9、11、5、8, 输出的结果为:12、11、7、0、3、6、9、10、5、8。-*/#include <conio.h>#include <stdio.h>#define N 20 void fun ( int * a, int n ) int k,m1,m2,max1,max2,t; max1=max2= -32768; m1=m2=0; for ( k = 0; k < n; k+ ) if ( ak>max1 ) max2 = max1; m2 = m1; max1 = ak; m1 = k; else if( ak>max1 ) max2 = ak; m2 = k; t = a0; am1=a0; am1 = t; t = a1; am2=a1; am2 = t;main( ) int bN=7,10,12,0,3,6,9,11,5,8, n=10, i; for ( i = 0; i<n; i+) printf("%d ",bi); printf("n"); fun (b, n); for ( i=0; i<n; i+ ) printf("%d ",bi); printf("n");答案: 1). else if( ak>max2 ) 2). t = a0; a0=am1; am1 = t; 3). t = a1; a1=am2; am2 = t;第6题 【程序改错】-功能:读入一个整数k(2k10000),打印它的所有质因子(即 所有为素数的因子)。例如:若输入整数:2310,则应输出:2、3、5、7、11。 请改正程序中的语法错误,使程序能得出正确的结果。-*/#include "conio.h"#include <stdio.h>isPrime(integer n ) int i, m; m = 1; for ( i = 2; i < n; i+ ) if ( n%i ) m = 0; break; return n ; main( ) int j, k; printf( "nPlease enter an integer number between 2 and 10000: " ); scanf( "%d", &k ); printf( "nnThe prime factor(s) of %d is( are ):", k ); for( j = 2; j <= k; j+ ) if(!( k%j)&&( IsPrime(j) printf( "n %4d", j ); printf("n");答案: 1). IsPrime(int n) 2). if(!(n%i) 3). return m ;第7 题 【程序改错】-功能:为一维数组输入10个整数;将其中最小的数与第一个数 对换,将最大的数与最后一个数对换,输出数组元素。-*/#include <stdio.h>main() int a10; void input(); void output(); void max_min(); input(a,10); max_min(a,10); output(a,10);void input(int *arr,int n) int *p,i; p=arr; printf("please enter 10 integers:n"); for(i=0;i<n;i+) scanf("%d",p);void max_min(int *arr,int n) int *min,*max,*p,t; min=max=arr; for(p=arr+1;p<arr+n;p+) if(*p<*max) max=p; else if(*p<*min) min=p; t=*arr;*arr=*min;*min=t; if(max=arr) max=min; t=*(arr+n-1); *(arr+n-1)=*max; *max=t;void output(int *arr,int n) int *p,i; p=arr; printf("The changed array is:n"); while(i=0;i<n;i+) printf("%3d",*p+); printf("n");答案: 1). scanf("%d",p +); 或 scanf("%d",arri); 或 scanf("%d",p+i); 或 scanf("%d",arr+i); 2). if(*p>*max) 或 if(*max<*p) 3). if( max = arr ) 4). for(i=0;i<n;i+) 或 for(i=0;n>i;i+) 或 for(p=arr;p<arr+n;) 或 for(i=0;i<=n-1;i+) 或 for(i=0;n-1>=i;i+) 或 for(p=arr;p<=arr+n-1;) 或 for(p=arr;arr+n-1>=p;)第8题 【程序改错】-功能:求出在字符串中最后一次出现的子字符串的地址,通过函 数值返回,在主函数中输出从此地址开始的字符串;若未 找到,则函数值为NULL。例如:当字符串中的内容为:"abcdabfabcdx",t中的内容为: "ab"时,输出结果应是:abcdx。当字符串中的内容为: "abcdabfabcdx",t中的内容为:"abd"时,则程序输出 未找到信息:not found!。-*/#include <conio.h>#include <stdio.h>#include <string.h> char * fun (char *s, char *t ) char *p , *r, *a; a = NULL; while ( *s ) p = s; r = t; while ( *r ) if ( r = p ) r+; p+; else break; if ( *r ='0' ) a = s; s+; return a ;main() char s100, t100, *p; printf("nPlease enter string S :"); scanf("%s", s ); printf("nPlease enter substring t :"); scanf("%s", t ); p = fun( s, t ); if ( p ) printf("nThe result is : %sn", p); else printf("nNot found !n" );答案: 1). a = NULL; 2). if ( *r = *p ) 3). if ( *r = '0' ) a = s;第9题 【程序改错】-功能:从m个学生的成绩中统计出高于和等于平均分的学生人数, 此人数由函数值返回。平均分通过形参传回,输入学生成 绩时,用-1结束输入,由程序自动统计学生人数。例如:若输入 8 名学生的成绩,输入形式如下: 80.5 60