数学建模社区-数学中国

标题: [转帖]++C语言中常见错误++ [打印本页]

作者: solucky    时间: 2004-6-6 19:01
标题: [转帖]++C语言中常见错误++
<>C语言的最大特点是:功能强、使用方便灵活。C编译的程序对语法检查并不象其它高级语言那么严格,这就给编程人员留下“灵活的余地”,但还是由于这个灵活给程序的调试带来了许多不便,尤其对初学C语言的人来说,经常会出一些连自己都不知道错在哪里的错误。看着有错的程序,不知该如何改起,本人通过对C的学习,积累了一些C编程时常犯的错误,写给各位学员以供参考。) X" `( `$ c: q' V+ J' u
1.书写标识符时,忽略了大小写字母的区别。
" B6 [/ Y# u$ W$ f6 Ymain()
0 M# R' [% S2 R/ _9 S* M: E4 x# w- R{! j$ ^0 v9 ?: K
int a=5;1 j, T+ V  |  G! @+ J" T& D& [; K
printf("%d",A);
0 I1 Z  P! P8 u( I) Q" _/ u}3 A2 @& j! ^: \! p  R# _. H8 J; B
编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。  a; p: m; {. n5 a/ X& J+ J
2.忽略了变量的类型,进行了不合法的运算。
4 J4 j  j( A5 T( v, Y/ |% mmain()
  ?5 V2 v0 P& C  A{
5 t+ u8 M" k; ~# X9 r1 Ifloat a,b;
! \  v" S! @9 Sprintf("%d",a%b);
# S* C# F3 x& X; v8 y1 V; J1 w}
# N5 O$ [) m( D" N4 |%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。. R1 k2 G$ s. V  H5 [9 v
3.将字符常量与字符串常量混淆。2 S- n; T/ ]# f- R8 ?- {
char c;! b  W. K. x3 y3 _# c/ X
c="a";& K: T1 {2 O5 P% w+ ]' ~
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。
  o/ F- n5 I) y, A$ _$ J4.忽略了“=”与“==”的区别。( a: b; t- w2 A  [
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写
0 P1 @2 t  L( y- bif (a=3) then …
8 N! f' F+ G5 f  {! Y8 z# ]但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
$ q2 t% C7 F, U+ V  l4 p$ B4 g  I: J! hif (a==3) a=b;
. W% N$ G$ @' n前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。
; ~% e6 C* g5 V0 b$ `5.忘记加分号。
- b7 b/ K( W  r9 a分号是C语句中不可缺少的一部分,语句末尾必须有分号。2 `' b8 V9 [  L) i
a=15 v3 @2 b1 U; L0 _8 g
b=2
$ o' O( t8 a5 P( u编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。' l6 P: I. n% D* F
{ z=x+y;) ]! \8 J( T# W4 {1 {8 h
t=z/100;, n1 ~7 V/ h5 N0 |" T
printf("%f",t);
9 Z9 U. u7 \  ?4 r4 Y2 r}% [/ i. S0 @4 t* D
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。9 u. U7 m" ^; p* p7 N- g
6.多加分号。
2 s! s9 y4 I9 I2 ]! y3 j' S对于一个复合语句,如:
: u( Y, u' J. K' {! B' q{ z=x+y;
( A9 E3 U7 r  g+ S# T8 W+ J7 gt=z/100;% W; l" x! N1 v* R$ k6 [5 U
printf("%f",t);
8 c& I5 M* E0 A- E6 a};
  u: X$ G- z0 d复合语句的花括号后不应再加分号,否则将会画蛇添足。8 S- j; Y" P% w7 x
又如:" g( N2 O, n* X% l
if (a%3==0);0 n) t1 u6 ]0 U. F! C! A0 ^& F
I++;7 f0 J; l+ M( b/ |. q( Y
本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。0 p/ b3 i5 ]3 z. B% ]3 Q" B- f  ~( E
再如:
5 c' R  J. {7 L% T. Yfor (I=0;I&lt;5;I++);
1 y% s, j7 K8 \3 N{scanf("%d",&amp;x);& k0 e# t% b8 Q
printf("%d",x);}6 c7 m! V1 D. d  g5 p
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。9 D! i* p3 M. Y
7.输入变量时忘记加地址运算符“&amp;”。) F9 ]' V7 E  N# W
int a,b;  ^. j! q! Q# p/ V- S
scanf("%d%d",a,b);
- i6 O2 g& O  v' _' L这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&amp;a”指a在内存中的地址。2 `! Q( u1 D( w2 H) |" k* u9 y' `
8.输入数据的方式与要求不符。①scanf("%d%d",&amp;a,&amp;b);
% J9 T1 y5 N' i" U+ n, [1 c输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:1 [& A: `: b1 e( {( U$ J$ F
3,4 3 M/ ]  G. p  D
输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。! Q" F& f& `1 D7 @% }- W
②scanf("%d,%d",&amp;a,&amp;b);1 g) U! e4 B. _3 h
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:
* l& b2 D. c9 T, i% N3,4
2 o# i) r. L3 c此时不用逗号而用空格或其它字符是不对的。2 q, v8 W% ]: R. l6 K# _7 n/ K
3 4 3:4
5 E8 n3 ?9 Q2 R; j又如:) B3 Z6 }3 m( b( j
scanf("a=%d,b=%d",&amp;a,&amp;b);
7 }6 N' q8 t) u7 {; v  q$ w% [" K输入应如以下形式:
0 U  y3 s9 |5 [0 x) ~a=3,b=4
8 t7 I4 b% @: x+ D' e9.输入字符的格式与要求不一致。7 ^- M' u. Y! R9 }2 n
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。( U0 l7 B. h( K  i# _" _
scanf("%c%c%c",&amp;c1,&amp;c2,&amp;c3);
- f% ]5 [, m$ C) j- l; b2 i% h+ I如输入a b c # u% W  h% Q6 [: H1 U
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
2 h5 Y! O- @. N) l5 T3 t7 c% p$ n10.输入输出的数据类型与所用格式说明符不一致。) `+ H( \. O; O. j! C' i
例如,a已定义为整型,b定义为实型
* y0 d. x, m* w: N8 T- _a=3;b=4.5;
* v: O% F, e0 K7 qprintf("%f%d\n",a,b);& [" w. S; [/ z! c' r
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
6 G" a3 V5 p5 ?11.输入数据时,企图规定精度。
" q! T2 f/ |7 |0 z9 K& E% oscanf("%7.2f",&amp;a);
& J3 z" q, F1 }% O" G4 m0 `这样做是不合法的,输入数据时不能规定精度。+ _+ B- C% o' T) H- f2 y( N/ X
12.switch语句中漏写break语句。
1 g, i, K) F! D' e例如:根据考试成绩的等级打印出百分制数段。0 J; X2 A, H7 @0 F8 M" S
switch(grade)
" W& O7 v+ Q8 J/ ?$ q( i9 l{ case 'A':printf("85~100\n");
. C5 q7 }4 C$ j3 `case 'B':printf("70~84\n");2 c$ d8 z7 y2 u
case 'C':printf("60~69\n");
" b% P/ V$ E, b  ?5 R5 d& Vcase 'D':printf("&lt;60\n");0 o8 K6 P+ h$ i' e1 L9 X9 E9 S. _
default:printf("error\n");
5 p6 s8 F) u3 G) U1 n% i1 A1 D由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如
8 {) ^' z+ |0 y' R- b- h( a- Kcase 'A':printf("85~100\n");break;
9 H) s' V7 T$ J+ m% _13.忽视了while和do-while语句在细节上的区别。
# L6 b6 V- u* s$ P! I' E(1)main()
) q3 Q, ?/ g1 g6 W{int a=0,I;
9 W! ]. [2 X/ O! @scanf("%d",&amp;I);
- E- h1 M4 Z1 cwhile(I&lt;=10)* a' r5 W; V  M2 S( H# Q
{a=a+I;
$ }& y) S& R% ?4 T- s2 e+ J% i. i- EI++;
) A+ \4 Q3 S& P7 L$ |}& s, m9 a  L4 s- }4 w& K
printf("%d",a);
9 P+ x. `& L  r# O9 E5 E}
& {8 X! L6 R& J) {2 k: U5 e: Z% G(2)main()
6 M6 ^5 N  m: X3 a{int a=0,I;4 v( M' H6 t( u, Q" S, t$ q
scanf("%d",&amp;I);
: [  T. M5 d, A/ bdo* ]8 b9 a& e) A0 i* Z- F
{a=a+I;6 |  E* B* R6 ]# |/ G% I% ]
I++;
5 O0 {& w# B- Q+ o}while(I&lt;=10);
  d% ?, ]8 t/ o' T* C9 e. Iprintf("%d",a);) g1 p6 u/ a" L8 h- |0 U( ?
}* e2 `5 k" A, _( d- A7 d
可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I&gt;10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。
- a8 H/ p' P# {. |9 Y$ T+ q) X14.定义数组时误用变量。
& d1 K8 b, N' i# Z) s0 aint n;
* p- F; \/ K2 l9 x5 m6 y1 Mscanf("%d",&amp;n);
. `: W2 g$ h  a6 Q6 g/ Xint a[n];* M* l7 u. C1 `9 G; I
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
) I0 i9 _2 t/ p9 u+ K! Y15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
- V! i# k/ s6 g, Z4 X! ?* X& v+ X& K9 P- }main()
% V1 C8 A: c' w  r9 Y' a{static int a[10]={1,2,3,4,5,6,7,8,9,10};
5 U4 T9 ^2 L' G6 U1 {printf("%d",a[10]);# _% j5 i! }0 B, q/ `3 t9 X
}; p" e% }2 Q4 U& c  N( U
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。0 ~/ j0 W- @* T+ d2 F
16.初始化数组时,未使用静态存储。: Y2 \7 ~4 c3 q
int a[3]={0,1,2};; G" N7 w. M3 I3 m' ^) b+ w
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:4 T7 M' J- R( Y- N7 ~0 _
static int a[3]={0,1,2};* ^( u! e# J. o: L- S" d
17.在不应加地址运算符&amp;的位置加了地址运算符。6 b4 q' g7 t9 |4 t0 f% y* R4 t
scanf("%s",&amp;str);" c5 H% C1 V, r3 D5 T0 R& _0 A
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&amp;。应改为:
( P' a" V5 h0 Dscanf("%s",str);' s3 ~; u7 b4 t6 Y% x* b2 S) n
18.同时定义了形参和函数中的局部变量。4 l9 [2 k' I4 a
int max(x,y)
5 }- [7 b& h/ h; J8 G. jint x,y,z;
1 N7 R# U* C7 v{z=x&gt;y?x:y;
* K1 j; x) A$ j; {1 \/ T0 y1 z2 S, R+ Areturn(z);6 [  a+ x2 }# G3 s; O
}
# `, L- Q, z. o% G' @2 ?. C! U/ Y形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:7 }! I  E( y1 j3 r8 n: S( J
int max(x,y)8 p6 o: k( b& B2 P/ J
int x,y;& M6 d5 N1 B& ~3 k% A& K* \+ v
{int z;4 A+ I, i' m3 q" _
z=x&gt;y?x:y;, I# E7 f0 _! _' S
return(z);% i; m& n. \$ E. s( O. n
}</P># M) T' H+ r: Q- n7 D" W+ `
<>以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P>




欢迎光临 数学建模社区-数学中国 (http://www.madio.net/) Powered by Discuz! X2.5