- 在线时间
- 0 小时
- 最后登录
- 2006-3-12
- 注册时间
- 2004-5-30
- 听众数
- 1
- 收听数
- 0
- 能力
- 0 分
- 体力
- 262 点
- 威望
- 0 点
- 阅读权限
- 30
- 积分
- 109
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 27
- 主题
- 17
- 精华
- 1
- 分享
- 0
- 好友
- 0
升级   4.5% 该用户从未签到
 |
< >C语言的最大特点是:功能强、使用方便灵活。C编译的程序对语法检查并不象其它高级语言那么严格,这就给编程人员留下“灵活的余地”,但还是由于这个灵活给程序的调试带来了许多不便,尤其对初学C语言的人来说,经常会出一些连自己都不知道错在哪里的错误。看着有错的程序,不知该如何改起,本人通过对C的学习,积累了一些C编程时常犯的错误,写给各位学员以供参考。% d" d- }$ n$ H3 b
1.书写标识符时,忽略了大小写字母的区别。
( c9 H4 j7 i1 cmain()
8 a4 R5 i2 ?/ `' {: m$ g O" e{
! }7 f6 ]' P/ u% |int a=5;4 v- i" @4 ~. d P: a) n! M5 }- c
printf("%d",A);' i+ t! z) l' R# X5 u& U
}
( W+ A4 X" d1 s- o4 A* {; |4 C编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。& ]2 ]1 p: W9 Y8 }& [) ]
2.忽略了变量的类型,进行了不合法的运算。) `: B1 S# |+ H& ?+ ]
main()
% ?& C5 d4 @. n! e{
6 o* x- S4 M, o& ?, g2 A" ]float a,b;
) z" G, \8 i& A8 Hprintf("%d",a%b);
# s4 s. E; \3 t! w q} u. s5 f/ R2 ?6 c9 W7 w/ u
%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。
/ N4 G# J( Z" C3 V: L" o6 [3.将字符常量与字符串常量混淆。% |0 F! F( \* C& M1 }; \5 a
char c;
1 l. C* z# M! [( K2 zc="a";$ Z# T& C8 E5 w
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。3 e& w% l! `) y. v4 W" M$ ]
4.忽略了“=”与“==”的区别。
% `+ {" c7 b% g: E6 M4 m在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写: i+ q( ^% C$ ]; E. z+ b
if (a=3) then …
1 H3 b( H- ~8 Z但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
9 {- d! Q# f6 e4 j# x4 Wif (a==3) a=b;
* U2 }$ H- D- b( c' G$ [- O前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。
5 x# J' r) m2 O! e" e" b5.忘记加分号。) n l- A! i) J9 w) H
分号是C语句中不可缺少的一部分,语句末尾必须有分号。( Y, B L7 @ d( d* E, \1 f
a=1
4 C+ O& b0 x+ A. x* ~8 |1 jb=2
3 S# c8 [2 B: a$ U) Q' m2 d" Q5 a8 o编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
+ z4 _/ E7 j6 {3 R. \2 T{ z=x+y;( I& a2 N* A+ Z. A8 `" B
t=z/100;1 Q _) r8 V- _
printf("%f",t);
! [& Q( m0 @: [' `. R& q}, t" z' A* O3 r/ r/ K8 [5 \, c: R
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。% q0 }' D3 Z" }2 h6 x; N
6.多加分号。
- b+ b1 k/ U' G! \, m, D对于一个复合语句,如:" @9 e0 t/ x1 ?
{ z=x+y;1 a6 F! X/ c/ v7 d: ]6 D ]4 U9 ?
t=z/100;
, \; J) u' G$ x2 l/ uprintf("%f",t);
* C! V; J8 R( I o/ R A, C};
3 X/ s' N4 p) M' i5 w9 S复合语句的花括号后不应再加分号,否则将会画蛇添足。
9 `1 x) {* D$ }* [9 D1 o( M又如:3 A* G# F' o8 d2 m% `! S& ^, N
if (a%3==0);* n3 K4 g& o) Y" s
I++;
h6 u8 C: E) ?9 l: |本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
x0 t& J7 D% t& }2 \& R" h再如:
8 L. S) o6 d, Z$ A" q6 T; }3 Xfor (I=0;I<5;I++);
4 t0 ]: N+ }* d( U! O* q$ C7 d{scanf("%d",&x);
4 P, |, H- t/ |3 R8 c, v% jprintf("%d",x);}
0 A0 k1 |8 U/ c本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。
# N# Q0 p( S% @/ O- Z" e$ `7.输入变量时忘记加地址运算符“&”。
5 d5 e: l( k3 Z. cint a,b;8 c+ g$ R; x( u# l
scanf("%d%d",a,b); \$ K$ N S0 T2 X8 ?& e5 E
这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。# P! I7 E& {. t3 a
8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);2 b/ q+ K5 n% I. Y6 F
输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:
) Z" O* |9 N- f8 Y2 I: V8 f3,4
( U5 d8 h1 l/ Z: Q输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。
1 e- K: n; c, k6 K②scanf("%d,%d",&a,&b);" @& Y! q% U" e+ S1 N
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:8 X% M& j' j: \5 p* s
3,4 6 P' c0 b* j2 c; b3 {( ?
此时不用逗号而用空格或其它字符是不对的。
+ e; j2 A8 P2 r5 Y5 @7 l3 4 3:4 2 j! }6 Q; Z6 ?3 Y- _
又如:$ Y# X) Q Y# e! d: T9 J a
scanf("a=%d,b=%d",&a,&b);
- M3 u, q o4 m% _- j输入应如以下形式:) X6 o! O' a s, t$ e( k% H
a=3,b=4 6 \' C7 R' r& p
9.输入字符的格式与要求不一致。# Y5 D- E6 x _, z- F
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
. n0 C3 X3 I9 n( Sscanf("%c%c%c",&c1,&c2,&c3);& s' w+ v ]( L' m
如输入a b c * P; X* O& m( m
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。$ {& f7 ~3 g- D; I1 ^
10.输入输出的数据类型与所用格式说明符不一致。
& P5 j/ ^, v0 Q* w" {例如,a已定义为整型,b定义为实型! Y% r8 |3 l; c8 ]. A# e3 m
a=3;b=4.5;& V) ~$ A) U8 ~0 v0 V/ D) D; m
printf("%f%d\n",a,b);
! R0 E2 Y, J8 _; x$ \编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
4 u* b ]7 i1 ^1 ^3 Q11.输入数据时,企图规定精度。
, Q' M- U6 \6 xscanf("%7.2f",&a);
0 y) O6 t# }" U' }+ E. W. S1 H这样做是不合法的,输入数据时不能规定精度。! [- m/ w M% u! M9 j
12.switch语句中漏写break语句。
4 O: T0 a# v6 S9 h4 d9 E2 N" q例如:根据考试成绩的等级打印出百分制数段。
5 c& @8 \6 r* Q3 pswitch(grade)
, i K* B9 @$ C{ case 'A':printf("85~100\n"); ?8 l/ S0 U9 F; v' M
case 'B':printf("70~84\n");
0 D# Z c1 \, Z' acase 'C':printf("60~69\n");
, y5 e5 u- l& ]4 Lcase 'D':printf("<60\n");
/ }1 p& [9 S* O! I* Odefault:printf("error\n");, e, ?4 B. u, p/ _
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如
6 V; T9 r* L0 z0 \) w; \case 'A':printf("85~100\n");break;
4 g5 Q3 g3 j9 [% d13.忽视了while和do-while语句在细节上的区别。
1 R! N! S3 W0 s' z3 a(1)main()
- Y. ~+ t" |' f# U# H{int a=0,I;. q# P7 p( A l* ]
scanf("%d",&I);
# Y4 l. H8 P2 g% `) s' \while(I<=10)
; [! w6 \% D7 ?{a=a+I;
; O& r8 q/ S+ w3 a1 b" @* fI++;
' \$ W q: `; V% ^# B6 w8 A% E}
) @* y4 X, r \0 [printf("%d",a);# ^9 z# O' J) Z: G, s8 u& e1 `
}
# y: F+ F; ]2 l( H4 W1 e" f(2)main()7 F" ]" S" E7 i7 k( L8 m1 D
{int a=0,I;6 p# M( @; ?6 R& Y/ @ l! x8 T3 x' k, T
scanf("%d",&I);, b+ [2 A8 z7 f* e% H, h
do
0 `2 c* l9 x' ~. Y# z2 |{a=a+I;& t6 o/ ~; |0 o: M+ }- E
I++;
1 u9 T+ U. E h! X1 I% `5 g2 u2 h7 Q) X}while(I<=10);2 q0 r7 ^& y/ f* C; Q, A9 H. ` W" _9 i
printf("%d",a);
# z. O( B/ l' K8 A}
8 `% l2 W. K3 t- D8 v1 M* v可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。( F u( j9 ^8 |, e2 s2 S, x
14.定义数组时误用变量。9 F9 }3 G$ x% L$ S! y' y
int n;2 ?6 _% |9 L- B7 g) }5 N. g
scanf("%d",&n);
) D2 }2 ?! x% h s: A1 d6 z' Mint a[n];
( K2 }! F( W# G3 H4 W数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
' ^0 a# Z2 u- ~: G+ V, c7 f, X15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
4 h" A- ?$ T' S9 Z( F ~, mmain()
% p) C% e- _2 ~3 v J$ i @{static int a[10]={1,2,3,4,5,6,7,8,9,10};9 q( G, N/ V$ O3 [4 S8 D) }
printf("%d",a[10]);) p5 _' |; A$ v Y
}' _$ K$ `: H# c# X
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。/ J* r: T/ S. @; o0 _* [% J
16.初始化数组时,未使用静态存储。$ V4 P' x) F( V+ ~+ Z( F( p
int a[3]={0,1,2};
5 e' V1 u7 i- J: ~3 K& w这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:& g2 P7 U% ~% I7 j4 Y' B$ E
static int a[3]={0,1,2};
: h, Z0 k( [' h- m17.在不应加地址运算符&的位置加了地址运算符。) e5 G. \ A6 r6 `' A
scanf("%s",&str);. n0 I: d) J: c: q; g. ?
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:- L/ R8 V* r2 a; Q# F( `) ]
scanf("%s",str);
3 b2 x8 v7 w2 S- i F6 l% P' I# w18.同时定义了形参和函数中的局部变量。
. k8 ?, r& g$ I5 pint max(x,y)
! s M5 `8 _# H( k. X3 i1 h" X3 yint x,y,z;
# {9 X5 c g* l$ V3 L{z=x>y?x:y;/ N2 J7 ~, a9 j
return(z);' x8 P/ G" g, e
}
6 U: f) w- X2 L$ J形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
/ y% l) [4 L* m) K1 Rint max(x,y)) ~' B( `2 P' F- r
int x,y;
. @5 P: y4 I, ~+ A{int z;
, X! v7 T# q: I) d" p' ?; B- Rz=x>y?x:y;
3 M% D* N* v. C; l7 m9 Areturn(z);
6 b) y! O/ L- v/ x& J}</P>
* u8 Q" t: {' M0 Z) M& V< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|