- 在线时间
- 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编程时常犯的错误,写给各位学员以供参考。5 ~8 S1 x+ p/ X7 u
1.书写标识符时,忽略了大小写字母的区别。
' K. U' ?2 c" H3 T( ~1 x1 y R7 B2 jmain()1 M! x) ^7 P0 h& d& }0 W
{
) O! t- V1 `/ L% q4 h5 w; {int a=5;0 u. ]1 |- ^! J& r5 c5 r
printf("%d",A);
$ f, ~: y! A. C}
! O, c: Q: U1 P0 |. O( v编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。+ L: c* e' S& p& s" V
2.忽略了变量的类型,进行了不合法的运算。" k {; _. O8 k/ V
main()
J* q. Z- ^$ [$ o{
: g7 M. d6 k4 k4 m# I+ N7 W1 u6 Qfloat a,b;
. _$ j# j! Y" ^" O6 z4 wprintf("%d",a%b);+ Z# R' Z. O" s- {) K' i
}
9 V' r; j6 B$ R! |& s7 [%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。, N) |- O8 M2 y" x" m
3.将字符常量与字符串常量混淆。/ x# [; b# b- C
char c;1 Y( L5 ^* x o" l8 H+ }
c="a";
7 Q! d) A2 W! o& v4 Q- L6 w在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。
8 v' U# A A, r7 X2 `4.忽略了“=”与“==”的区别。# T& G- ]! g( r) R
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写7 d, a7 C6 l7 u
if (a=3) then …
2 x; a$ L) w3 @' n但C语言中,“=”是赋值运算符,“==”是关系运算符。如:7 S4 o# X9 z; }3 X* h8 o9 r
if (a==3) a=b;% p1 W. @ @' p8 U, ^0 W$ Z d
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。/ U2 v) [; D0 A1 d+ z
5.忘记加分号。8 g2 z/ }+ m6 `& v# e
分号是C语句中不可缺少的一部分,语句末尾必须有分号。 O: E$ Z! N, X% ]9 m
a=1
% P% Q. D* \# B' \$ X. m3 ub=24 P1 e& ~ ^( x6 d: G
编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
1 q' B+ K! |. k" m& K2 S{ z=x+y;
3 v" |5 r3 W |t=z/100;
) w, C! Q9 R" y% D: S- Fprintf("%f",t);
. n$ o# V+ G; k}
6 |/ z1 R$ \6 a5 e5 ^对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。
# Y; ~. F3 ~& z! G f6.多加分号。
# o6 D, W5 C+ L% v对于一个复合语句,如:8 E( e2 ]8 T0 p; K
{ z=x+y;
/ t7 K/ A& M$ k5 T. w Gt=z/100;
6 ]- Y1 w7 z4 S6 `9 B$ n2 sprintf("%f",t);& C9 ]# g% ]/ c8 _2 D5 S
};
- u' c+ y9 `8 Z+ \1 {$ L复合语句的花括号后不应再加分号,否则将会画蛇添足。; F( V3 o. M$ l5 g9 c
又如:
3 q" W7 j: a; _. A0 G8 Aif (a%3==0);1 F+ D. x' h7 f7 J. B/ n/ m
I++;9 F( i! O; O+ E7 J7 [
本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
. T3 G' `! r& h再如:+ M2 Q5 v! q: D( [5 {/ P( g' ?
for (I=0;I<5;I++);
* K; c) j/ E1 e9 a' g, A' D# E. }{scanf("%d",&x);
. q( K" R( W, b& I; ~6 bprintf("%d",x);}9 c7 u+ e5 n9 X
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。6 Z3 `( T: A7 ~% h; C% I6 L
7.输入变量时忘记加地址运算符“&”。" r3 x( _3 Z: G! U9 j- g
int a,b;
$ l; R3 s2 | d8 Oscanf("%d%d",a,b);
# U" e" ?1 H* g0 w( _这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。, s3 \' P% y- ]& V! C2 L4 b1 P
8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);
/ W8 }! y) t3 Q, c输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:! F; {8 T$ y# i) ^4 V* ^& J
3,4
0 N4 y9 w" J; `8 v _输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。4 V4 {7 I8 B+ i) a" }$ c
②scanf("%d,%d",&a,&b);/ s6 }$ C6 w3 L. _8 d9 _( {
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:
2 q5 Q* d8 f: ~! j+ i; l! O3,4 ) }4 t' O" y. u; [& d; Y- a
此时不用逗号而用空格或其它字符是不对的。
- f A6 H( j' b" \) ~3 4 3:4
/ V: o! p, F; e/ Q/ l3 j$ Z又如:3 O& T8 F. M; K2 F
scanf("a=%d,b=%d",&a,&b);+ c/ _& W' [/ K8 x5 S2 S% k
输入应如以下形式:
: \, w0 p" v4 ja=3,b=4
4 M0 V9 J7 D/ x9.输入字符的格式与要求不一致。
, F& I7 Z* v+ K: U; J. u在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
3 @' j" \5 d% W4 e& [% P/ [scanf("%c%c%c",&c1,&c2,&c3);$ x$ o/ I; x% W2 U u
如输入a b c # R8 X# e i: z: B2 d# z- x
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
& a$ ^9 R) C, E2 w( P10.输入输出的数据类型与所用格式说明符不一致。" E5 d) H0 O: ?
例如,a已定义为整型,b定义为实型
; G% U! U( }/ @/ za=3;b=4.5;$ `7 {' B' e7 y0 k
printf("%f%d\n",a,b); j6 K' X3 e- ~# [6 y7 P$ f
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
# Q3 s! Y. j. t* R; R& Z+ b11.输入数据时,企图规定精度。
9 X( g% e5 }2 y3 pscanf("%7.2f",&a);% k6 G" Q/ q. P- D5 D
这样做是不合法的,输入数据时不能规定精度。6 [: q7 }) t3 T" X0 ^, _/ D
12.switch语句中漏写break语句。/ z, W5 D5 D& g7 P
例如:根据考试成绩的等级打印出百分制数段。
* o0 ?, o8 q: y" t& hswitch(grade)
) P4 X: E+ G' I+ h7 Y; }{ case 'A':printf("85~100\n");2 e: E! c' d) J7 b6 `
case 'B':printf("70~84\n");
5 _, U5 v. c, F8 L7 h. p3 F* {case 'C':printf("60~69\n");5 [3 P. l% L. ]0 R7 N+ \! b
case 'D':printf("<60\n");
' k0 M# T0 B- F. O- z3 Zdefault:printf("error\n");4 Q( r# R* P" r$ m9 u" k
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如, q5 z9 D# U* F6 d
case 'A':printf("85~100\n");break;
$ G( p1 V; {' u( P; F# k13.忽视了while和do-while语句在细节上的区别。
# [! q7 x5 H# Z' P8 B- k6 }(1)main()
0 }# I9 G! i% Q' X9 C; B{int a=0,I;
, z' V& O. P: R ^. T* M gscanf("%d",&I);. s2 j- ]9 v# I" h
while(I<=10)
3 P6 ~ W7 e; Y- l{a=a+I;0 _& x$ f0 [9 l
I++;
7 r0 U$ Q, s# ~3 C9 a} s. g0 V* W" M+ c* |& [" _
printf("%d",a);
1 U# U( c% a& Y; L: j7 Q' S}
% H( u$ E3 h2 P4 Q' [, A(2)main()
& a0 |9 x' q: a. g* K8 d{int a=0,I;; G! R! g. z* W* S
scanf("%d",&I);- J) ^! k0 U0 _
do
) H* Y3 j$ X. ^2 Q* G{a=a+I;/ _3 t3 T: X9 T7 }
I++;
; Z3 T: D* j+ N, v}while(I<=10);+ V' R, @: b' t+ O
printf("%d",a);
# @. P! F0 ~$ f8 F9 B: M}
& L: e3 `1 k$ ^ h8 d" U3 y可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。: U$ ?0 Z* C# \
14.定义数组时误用变量。
/ ~- B5 h* y& F7 t; H& A. wint n;
9 l- l, l# \# x N& Q9 Wscanf("%d",&n);/ j" G* Z" ^8 k' r8 D
int a[n];1 p: F2 U5 W# \9 P [& w7 p0 P' |( J
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。5 i& s7 D7 N7 _% G
15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
5 [1 x/ Q @! y, [! S( l8 Lmain()
5 B; D0 d* Q4 `{static int a[10]={1,2,3,4,5,6,7,8,9,10};3 w0 O$ P% Z. |( T2 s. j( D9 t
printf("%d",a[10]);
9 `) }# V% z: ~2 J% J0 f" G}
9 ]6 l. l8 Q" [: a. L; {- ?( ?C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。
: Z% J( n. D8 M0 ?4 P16.初始化数组时,未使用静态存储。
- I$ d% N; r( C( s% k, eint a[3]={0,1,2};" J5 f, i4 {8 q8 U: o3 }$ x
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:
+ D) z: `/ ^& l9 X q/ _static int a[3]={0,1,2};! E' z0 v& y% ]* s
17.在不应加地址运算符&的位置加了地址运算符。0 }$ ?# R1 z4 c% g+ ?$ s" N8 T8 q
scanf("%s",&str);& H6 p2 p1 T& v x) M( B
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:" \1 }. T p8 ^- g, }
scanf("%s",str);
# w" e* |3 k: }' Y! o18.同时定义了形参和函数中的局部变量。
! C/ [2 P. M" O- k2 k0 bint max(x,y)4 I r1 S1 U( ]/ a. c, c$ A
int x,y,z;
5 P+ X8 r" `1 e" ]{z=x>y?x:y;
4 k1 @: j+ Y- D+ E" Y* o Nreturn(z);4 h* Q1 v4 [) `0 G! R: s
}2 e- R8 t M2 j; S# o3 S' w7 Z r% @* U
形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:+ j$ G% f8 i& h$ b- Q& {
int max(x,y)
. N* x7 O5 _. k" S3 k! iint x,y;
; P! n7 L& Z# B$ k* Q* n y/ [{int z;
3 k8 n$ f5 \+ A9 R/ Kz=x>y?x:y;
& K4 e/ D: H5 k5 h$ w) o# T/ v( ireturn(z);7 f" G/ ?( i3 @% V
}</P>4 }5 f( G( m- o
< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|