- 在线时间
- 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编程时常犯的错误,写给各位学员以供参考。0 T! s2 _7 E6 X% k& m
1.书写标识符时,忽略了大小写字母的区别。
! G5 _7 }2 \6 S9 i% h6 N: V2 W/ ~main(): K6 M: _% |9 g1 m$ |5 ? ~& v+ E+ {, w7 C
{
{7 x( Q* v8 }int a=5;( [: f7 q- W( h8 j4 |/ D
printf("%d",A);, O; u, v3 E2 F3 r# z
}
$ c2 q0 I% Y2 |9 r) H+ ]0 [$ k" i编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。5 g/ ]) \+ t. B& g. a
2.忽略了变量的类型,进行了不合法的运算。 N& P2 h. E! Z4 d5 J# o
main()% k3 I7 g" B, m& G
{
" M" n% N% z6 O" ?$ O6 y! p7 s& X! dfloat a,b;
3 {6 v& f" D, W8 `( jprintf("%d",a%b);+ D- Z9 f2 Q( N4 \ e$ C
}
& j: c0 }0 G7 |% Q8 s3 S8 R%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。. V1 t4 {0 j& `" q, \, `
3.将字符常量与字符串常量混淆。# E6 Z0 [ A! v) A, t5 i6 N
char c;
1 J4 Y0 f9 g4 n/ `& q' v# Hc="a";
7 W% X; O* {( ]0 \* S+ R在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。5 m( @4 L d8 Z4 U) W7 c
4.忽略了“=”与“==”的区别。
" `3 U% w' y% S) y2 |$ |) }/ S在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写
^/ ~8 P2 v# O; k/ V& }if (a=3) then …. z+ ^" O( S3 A" {* f* Z/ M; v6 `
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
, q9 L2 c' Y9 S& `! }$ ~" qif (a==3) a=b;2 w4 u" X5 |- Z8 S
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。. f0 Q( @/ ^& L4 u: z# _, L9 ^ w4 i
5.忘记加分号。
) U2 O( u1 u: v1 ?" i分号是C语句中不可缺少的一部分,语句末尾必须有分号。. S" X" T( Y* D) C: j* p* v
a=1
& I5 o% D2 Z8 l! b! E' U7 d& _% b" zb=2% S* F) D4 k" j! Z/ b6 W- {, G! I& t
编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
, I( w& z: p2 S1 ?$ y: C6 `{ z=x+y;
6 }( w0 z) B( m$ u( R$ {4 S8 Ct=z/100;2 ]$ O% S7 A1 F ]+ w: X
printf("%f",t);
8 t* j7 j8 X4 V* _' f- S1 A}
. O5 t$ W3 N4 k: Y对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。7 f6 A) i' r8 w1 ?- p$ |8 d
6.多加分号。5 i) }) U) ~7 @$ u7 B; {/ y
对于一个复合语句,如:
, F; ^2 v, e7 w: M{ z=x+y;
) r+ {, g7 ^, dt=z/100;1 w4 E! J! E' f; \, Z
printf("%f",t);2 @; J% Z Z; {8 P* }4 e N
};4 A# g' q+ I+ ^5 F7 e) y
复合语句的花括号后不应再加分号,否则将会画蛇添足。
+ z( }# @1 {' a. l' c2 m- @/ r又如:
+ X# S) D! ]9 [+ r8 f" ^$ Cif (a%3==0);
3 v4 `0 o6 V* F( w+ HI++;
' J; ?, k H" e2 r/ `! @+ F本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。6 N, L! d1 D7 {" V0 q6 o7 q8 F
再如:' k0 O: c1 Y: `" h
for (I=0;I<5;I++);
# B. e9 g3 y8 }& x{scanf("%d",&x);
7 b/ ]6 h6 {5 L7 Q: Aprintf("%d",x);}& v- T7 ]1 x7 b& ]
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。
) G+ Z! ?7 x% J% d/ h/ l) t$ W& a7.输入变量时忘记加地址运算符“&”。
& v' g0 Q4 Z0 i5 K5 Cint a,b;( ^+ j+ V% |) x, t
scanf("%d%d",a,b);
/ @4 Z- [& [( G# A) {; S, X这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。
. i8 Z$ _: c/ ]7 U# V8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);
% v/ z p* [6 j9 R9 o) n4 h输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:
2 W3 o3 T2 m' Y" H, B. ?3,4
/ p2 r* w" ^* E, V5 x; b! m" ~: R1 B输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。 ? n, y4 ~' a# P; W+ R+ [
②scanf("%d,%d",&a,&b);
3 u3 S# }4 q0 |C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:1 x- |9 v7 A- I
3,4 $ e6 q6 P+ N( Z! b1 q
此时不用逗号而用空格或其它字符是不对的。
. i0 x5 w2 l* W, Z( v" j5 q8 Y3 4 3:4 6 b D: [ O! v/ c
又如:
& u6 a; v; Q( k9 ^# ]: {* o0 Cscanf("a=%d,b=%d",&a,&b);
; h+ s& l. T. M! a输入应如以下形式:
+ G4 D2 l- P2 d8 E5 Fa=3,b=4
5 s: E! l/ L0 ^3 X; }' s3 F4 R9.输入字符的格式与要求不一致。+ M8 S. w$ p4 A3 }
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。. M) C8 c9 n2 D0 E% d. s! y
scanf("%c%c%c",&c1,&c2,&c3);
! x) k2 I. G5 J4 M( c如输入a b c
% o5 @( y* ]+ }1 \% q* h+ i字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
. x% L6 u. i5 \) e, t# e10.输入输出的数据类型与所用格式说明符不一致。
2 P) _4 ~2 p; ]2 O例如,a已定义为整型,b定义为实型
" i" Q# Q; ~* V! s. j2 Ca=3;b=4.5;& V9 J5 c- }+ ^2 s
printf("%f%d\n",a,b);! P7 K5 t+ H& g6 H u. t
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
M8 N+ c7 ?; d11.输入数据时,企图规定精度。5 b) Q+ t5 l7 F7 O5 J
scanf("%7.2f",&a);
. I* T4 A% @2 ^6 [- g1 D! a这样做是不合法的,输入数据时不能规定精度。
4 u: n+ B/ `6 [6 F: s12.switch语句中漏写break语句。! c! y" n2 w2 e* V2 ]& R! j4 G
例如:根据考试成绩的等级打印出百分制数段。3 n7 J% [, u/ }
switch(grade)
/ ]' X, _: `3 v" t7 ]! n8 t, j1 u{ case 'A':printf("85~100\n");' I0 D/ ~; Z+ R! M9 K' V7 l% ?
case 'B':printf("70~84\n");2 ^: \& k, N3 A
case 'C':printf("60~69\n");
7 b/ G1 n O, a8 Jcase 'D':printf("<60\n");
. E) A; }" S4 b5 t( Y# C6 o$ ~default:printf("error\n");
1 v% j* T- }) _0 p7 T0 J- I由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如# S$ S1 s6 l3 d6 ~/ p
case 'A':printf("85~100\n");break;! I1 T4 J* s- q1 N2 L
13.忽视了while和do-while语句在细节上的区别。
- b& X! Y$ i& ^3 ^3 d+ e(1)main()
- q* |, t1 z1 w: g{int a=0,I;
" z+ V( u$ A4 H% |( G& t4 _" O6 m& pscanf("%d",&I);: V4 R. b% | E" {! k
while(I<=10)
& _5 p9 r5 b6 g8 m& o{a=a+I;
+ j5 g5 x/ ~) P! eI++;
) z# M- @( s/ i, m( i6 @) G}5 f* i- x r5 H2 r4 H9 J
printf("%d",a);
$ u0 d3 v$ y0 L' U. l2 T2 y}% n' G" S& e* X" L- f
(2)main()# x0 m q7 K' L* t' A/ M
{int a=0,I;- ]( L! Q9 V2 O. V# m# d
scanf("%d",&I);$ m$ p2 n9 k% ?! Q/ z; P
do9 V3 q# M3 a. o% N: P( J* [6 \3 y
{a=a+I;
% s8 E" d+ e2 i: a- d7 C. SI++;0 @8 D! y9 P* ^1 q1 e+ l( D! \
}while(I<=10);3 u4 a. A8 N, j8 ^, ~. l5 n
printf("%d",a);, q9 w4 ~' D. E# ~& ^3 Q! T% K
}
; x1 f& ^: `5 q, N/ P; B可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。
8 X# c* t9 V( S/ A% D14.定义数组时误用变量。
; k) ~4 O3 S/ J8 n7 qint n;) I4 r$ N$ s* X; [) q
scanf("%d",&n);0 B6 q2 C' b( c3 i6 p! |/ x
int a[n];5 r+ ]3 A( t+ N" n: z Y" _
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
- Z; t) [$ m: s4 G1 e5 I15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。8 t1 @' B; M% M! R1 l
main()
- _/ H% M" S. ]{static int a[10]={1,2,3,4,5,6,7,8,9,10};
& i1 A" Y% ?% B# f) o3 C* ~$ O- S0 { Lprintf("%d",a[10]);: Q1 ~" v2 Q3 S, u9 l& V5 o+ _2 c
}5 }% Y8 U! B- Q, e! X
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。
: ~! B) g; r) U7 j, ?: f4 J16.初始化数组时,未使用静态存储。
0 j" _, U k: E3 y n: Y! }int a[3]={0,1,2};- @) J8 w7 }" N8 X( C
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:
8 @, A9 c B# q" {- Istatic int a[3]={0,1,2};. _* }! D# r$ e+ ?$ V" C3 c
17.在不应加地址运算符&的位置加了地址运算符。
) y: ]; [2 f: Gscanf("%s",&str);# d% Q* B0 |$ A8 q+ l
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:) r B7 T/ E \9 s' Z7 V
scanf("%s",str);
2 G; B7 x. H# ?5 X" q8 i18.同时定义了形参和函数中的局部变量。+ a% P7 `2 x! H8 x' l0 ?3 o1 Q
int max(x,y)
% r5 J2 d4 r* z7 T$ x; Mint x,y,z; Y+ t0 n) J2 T! H( I. ]
{z=x>y?x:y;
) A$ z/ q4 `$ Z1 V8 ureturn(z);3 k8 L* v2 p! ?
}
0 L1 a8 W6 K" i g0 R+ t形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
5 E) x$ b a6 r$ q, H Bint max(x,y)
$ i8 ~, o3 |! o# H" w) K9 I6 \int x,y;9 S( [# a# l/ S4 Y P: n8 h3 |
{int z;
( C8 [/ V. P2 Gz=x>y?x:y;; W/ ^% H0 Q9 z0 n; E
return(z);
* q5 `8 S1 ?+ v$ y. R; W# `}</P>
* R5 e' j0 a: o3 \: N( Q< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|