数学建模社区-数学中国

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

作者: solucky    时间: 2004-6-6 19:01
标题: [转帖]++C语言中常见错误++
<>C语言的最大特点是:功能强、使用方便灵活。C编译的程序对语法检查并不象其它高级语言那么严格,这就给编程人员留下“灵活的余地”,但还是由于这个灵活给程序的调试带来了许多不便,尤其对初学C语言的人来说,经常会出一些连自己都不知道错在哪里的错误。看着有错的程序,不知该如何改起,本人通过对C的学习,积累了一些C编程时常犯的错误,写给各位学员以供参考。
. f: V4 y) G- O7 \+ j7 v1.书写标识符时,忽略了大小写字母的区别。' H+ d  `0 b2 N
main()
( ?7 A* l+ z0 U{" s- a. e6 r: T1 p3 o
int a=5;3 ]& |" \# H$ k
printf("%d",A);2 j3 w  p* T- t& U* ?2 T) R. M8 B
}& `6 M! g( t1 m8 D9 y
编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
8 {& F9 V; J: U' s0 p2.忽略了变量的类型,进行了不合法的运算。
& U( t( ^- G6 c, H5 emain()
$ S- N! A# o, P& x5 w5 ?9 f% M{( e3 R% R7 n& o
float a,b;9 m8 n3 c7 T8 j2 t+ n% A
printf("%d",a%b);. U2 Q. }# ^+ T4 m1 `: N
}
* L4 `, K& b7 w$ J%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。/ `8 S: u5 T! B) C
3.将字符常量与字符串常量混淆。7 y% |3 o3 _, O6 a  e" Q8 M
char c;$ P% ?3 O) t. M2 N; Q
c="a";
7 c4 h# t, z: Z! n! q在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。
) t- c3 ~3 R; Q: Q! @4.忽略了“=”与“==”的区别。
& }& }9 D% s: s! s在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写9 {  Z+ }) F7 Y2 H; `2 s
if (a=3) then …/ G1 d) g# s' [+ R6 N
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:6 v8 j* l, b  `  O" q! [+ z
if (a==3) a=b;
! m4 K, J6 C) M6 a前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。0 H! C0 _6 a  Y( M. B
5.忘记加分号。) m; h! ^- q/ t# X. J7 g9 L
分号是C语句中不可缺少的一部分,语句末尾必须有分号。8 L$ s: c# ~7 K! F5 |' Q: u
a=1
" C# t% q! F. L6 k" }! G- Fb=2+ }2 ^- B# _. I$ w8 d  S% t
编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
- w: t# K$ ]. `5 V( X{ z=x+y;
/ i- G/ W) J+ Q6 tt=z/100;- T) q( _1 o+ m6 u! n7 M% y# r+ P8 ?- {
printf("%f",t);
8 W: Y! Z/ [2 y3 c}
8 q2 B6 M( w2 ^+ ~$ i' s对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。& Y; P8 D. O$ x1 k
6.多加分号。3 a; ~' _! k9 @* ]' q" p
对于一个复合语句,如:
; [. Y4 v9 Z4 P0 [; B! Z{ z=x+y;
, t1 @8 X' {) `8 {' _1 d. o7 a8 ~t=z/100;4 e5 c8 w1 Z) c
printf("%f",t);8 L9 X  i% N. g$ m5 o& G  \
};
+ A, I" k& b4 u+ W4 f$ H* H0 b复合语句的花括号后不应再加分号,否则将会画蛇添足。
5 b! q1 ?$ T, n- X) N, {又如:
2 G/ |# j$ T- x+ K3 y7 M7 h# ?- n0 Xif (a%3==0);/ T, |. d6 V8 d$ l4 z3 I0 F2 S
I++;3 f: {: y$ z* H# x
本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
/ j, J" Z5 j( o. u9 k再如:2 B$ I& I% S5 O
for (I=0;I&lt;5;I++);( q, J" {0 T: j8 F' N& F- r: b
{scanf("%d",&amp;x);6 E% U5 j2 R) z! R. Q! Y
printf("%d",x);}0 C* s2 y8 ^3 ~# X7 @
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。
! a& L; n" k! `" W7.输入变量时忘记加地址运算符“&amp;”。) C6 W# W3 }( S9 c  `5 B6 X
int a,b;" I4 j# G& e$ m4 ?+ a9 O8 s! J
scanf("%d%d",a,b);7 n* M) \$ d/ S" C% A
这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&amp;a”指a在内存中的地址。
# B1 {+ Y0 |: A! \) H7 F8.输入数据的方式与要求不符。①scanf("%d%d",&amp;a,&amp;b);  [! h6 B! U" C. a
输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:
4 n$ C% v# s. {: l" X* _3,4 & a8 T- @* m% _; Z7 ~+ ~. }+ K
输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。3 ^/ h3 j6 H2 Q) A, w
②scanf("%d,%d",&amp;a,&amp;b);
( Z: M6 w# g! U2 Y9 lC规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:' {* o7 Z. {7 z
3,4
) ?2 b' J0 F0 A) L; B. A7 w1 G此时不用逗号而用空格或其它字符是不对的。! \2 }/ F" |  O* g; I) B7 B
3 4 3:4 0 D/ C6 i  w" b! b' H/ {
又如:
' i# \8 ~, s! m' Pscanf("a=%d,b=%d",&amp;a,&amp;b);- E# o9 q5 A* J) g
输入应如以下形式:, q$ e) K* P$ L/ Y
a=3,b=4 ! l; F0 |2 W6 _/ j' C5 H2 r) a
9.输入字符的格式与要求不一致。- e1 j, w: J# }9 S7 E; l
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
6 N9 x% n: i; o9 y) Uscanf("%c%c%c",&amp;c1,&amp;c2,&amp;c3);
# A5 Z$ D% L/ k% B5 s. q5 ^5 ?如输入a b c
* m3 |3 P$ ^" n: d' [& A$ u字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
2 E, b$ l- W1 Q( y. T* H5 R6 t10.输入输出的数据类型与所用格式说明符不一致。
( A* J9 a* j  ^. S例如,a已定义为整型,b定义为实型5 i  ~; L. w/ R4 z4 b$ K
a=3;b=4.5;
2 z; ]; Z+ q* I4 X9 d# _3 sprintf("%f%d\n",a,b);7 V& K; s; H0 n8 {# z, x+ F4 k/ b
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。1 c% X: f% {  i
11.输入数据时,企图规定精度。
+ C/ A$ F% q8 \9 yscanf("%7.2f",&amp;a);2 y# a5 W4 Q% T; n9 l+ h7 |" h
这样做是不合法的,输入数据时不能规定精度。
" x6 _. [7 e1 N% B12.switch语句中漏写break语句。
+ ?) }( b" R% F6 n8 W9 W例如:根据考试成绩的等级打印出百分制数段。+ c% F7 `; ?4 S
switch(grade)
% n; K( ~$ d/ T2 `{ case 'A':printf("85~100\n");% E. r' S6 }/ w( M$ w' I
case 'B':printf("70~84\n");
) y2 U" f2 j& T5 t1 Jcase 'C':printf("60~69\n");; D4 d2 S# e1 [  Q. y3 n( {3 e
case 'D':printf("&lt;60\n");
, \" o6 g. ]. c  }8 l7 wdefault:printf("error\n");+ x5 l! i( \- ]: a' C  G* C
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如" I0 p  N7 ~, U! Z" o. `" F
case 'A':printf("85~100\n");break;4 Y2 [% }  p/ j& J: d' o+ l
13.忽视了while和do-while语句在细节上的区别。
3 b( [$ d+ a  @& Y  X$ i7 x: d+ }7 h(1)main()  |7 Z5 r4 k& G1 g! e7 m& a
{int a=0,I;
. B* r. n/ `& W( y: xscanf("%d",&amp;I);
, [0 o7 [2 `" s3 ?) N- V+ F+ ^while(I&lt;=10)
' Y. G" G9 m+ ~6 e" I{a=a+I;2 B8 B# ~# G/ V
I++;
1 J& ~2 _7 ~& Y  b. N& P}
6 x( t1 s5 y7 r+ t; Z5 zprintf("%d",a);
- B) o- Z/ K" c9 ?}
: y# L! i5 |0 V  u6 D: h6 A(2)main()
0 F5 |/ z  h. C7 s{int a=0,I;- a- b# g7 j, \3 U! k9 P5 n* ?
scanf("%d",&amp;I);- O8 |: s; g* G' K/ {# A$ o
do
! `% L0 n( Z2 D2 V" f{a=a+I;
, ]/ s, f5 C: g1 O/ \I++;5 o% K! g; d- v! B
}while(I&lt;=10);, @* ]' D7 z& ?  w# L  Q- D- ^
printf("%d",a);
. R7 h; X0 s% k4 k% D8 @* h% L}  q2 o' T2 B$ K, E! f7 @" C
可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I&gt;10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。( e6 Z4 t3 z* H) @2 s
14.定义数组时误用变量。
, |3 u! r$ U) Aint n;
0 k% a' N1 i) G  V3 c" v( |" wscanf("%d",&amp;n);- W  @; Z: j* y, s  |* B; G& J
int a[n];# `! W: D- r8 W6 z! P, O2 G
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。5 x0 Q, q% Q& d# b" \; g6 S8 N
15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
' t7 L4 x6 M. a+ i7 Emain()1 X( o) k6 h7 B6 @3 ~$ \
{static int a[10]={1,2,3,4,5,6,7,8,9,10};
- x  m& A5 c& f) \+ U  {6 Oprintf("%d",a[10]);
) W) R3 R/ E" I5 j1 _}
' }; Z9 V# Q& e1 B- gC语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。: D( o+ |1 x. i/ ?( g& L
16.初始化数组时,未使用静态存储。
0 E" W9 q0 O5 p' Fint a[3]={0,1,2};
0 W: k6 s, ?# g! ^6 P这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:
* g$ V& g' N' |- U" ?7 R: V; _static int a[3]={0,1,2};$ P5 h% i. k# J  g( N( e- A- Q! C
17.在不应加地址运算符&amp;的位置加了地址运算符。
0 t' q2 ?  C! fscanf("%s",&amp;str);
' c) i) @2 x$ N8 v7 PC语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&amp;。应改为:7 Q+ K* [1 N# {: @
scanf("%s",str);
- o) ~) \$ M2 w2 X, ]9 o0 X18.同时定义了形参和函数中的局部变量。
2 v3 W  X0 x! |  E; ?" kint max(x,y)5 G" `; O% ~0 R( ^9 D9 X1 c5 A
int x,y,z;
% E" l& s2 O. H4 y) u. d{z=x&gt;y?x:y;6 @! K0 L/ N6 C3 ]6 k) L8 C2 O
return(z);
: ~3 ^7 s* y2 G6 `& G. g}3 O- ]: T2 [6 Y
形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
0 }5 B1 Q7 [1 s) ?/ rint max(x,y)' |) Z4 Z* A8 _6 H3 p: D
int x,y;
$ O. c* T% b* `4 O9 S/ ]$ f{int z;9 N1 N5 @) D, c/ A
z=x&gt;y?x:y;
' U- U4 M  }6 P* R. L( Oreturn(z);
# T* G0 |0 c6 X4 z) i* Y  d}</P>
( C' m$ _; |. ]$ Z& n<>以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P>




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