- 在线时间
- 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编程时常犯的错误,写给各位学员以供参考。4 L- d6 ]0 }% _$ o
1.书写标识符时,忽略了大小写字母的区别。# T$ x3 v& ?2 R8 z3 s
main()
8 B; r- z" E$ ^{3 H! l( {: F. J# E) o5 V
int a=5;
5 l' X: |) h6 W( }, vprintf("%d",A);' A5 j# m2 ?& e) l: R1 j
}
' p8 u1 Z3 L. W/ v& D0 a! c编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
H% m; j4 t* k' M8 O2.忽略了变量的类型,进行了不合法的运算。% D4 y6 \6 V A( J! S! q/ q! u8 l
main()
# g" ^, p% f4 T5 b+ e4 ]1 y% F{
+ G) N/ }0 f4 w# R; Ifloat a,b; ~) b1 L7 N+ Y- e. n5 j' v4 \' ?
printf("%d",a%b);
* \8 |% [/ r1 {9 k( x D: @}+ t* S) C2 R; U: r$ r7 v/ z
%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。
* E1 E! B6 o% r! [; L: ]3.将字符常量与字符串常量混淆。
! {, ]1 ~; T* s1 R) f' G9 Z: nchar c;4 Y' q- _, m: F3 g. Q' C3 A
c="a";* ~# Z# B, U5 h' K q) D: E
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。8 u# c6 Z$ V! Q" ^
4.忽略了“=”与“==”的区别。) h }# |4 r5 J( Q( F' B
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写
7 B: L, J. h: Z- p8 y8 Oif (a=3) then …. y7 p' |3 a. J
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:0 ~9 q5 [; s3 W5 o t
if (a==3) a=b;9 d& q( r5 z+ J S/ U
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。
3 l1 c" V1 Z0 [' N1 E1 y5.忘记加分号。
3 [& R- W9 c& I, }* s3 _分号是C语句中不可缺少的一部分,语句末尾必须有分号。2 H& ^" g/ R) A4 N6 J! K" @
a=1" E, z; e( I4 J$ U
b=2
0 }" d' h9 Z2 T编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
& H0 j' ~1 v- S- E+ ]{ z=x+y;
4 J% r; ~% m! _/ |t=z/100;
! Y, A' ?/ K7 r; Z% L" [printf("%f",t);
% \- [+ \, @; n' m% T4 F}& ]: y: [9 c+ v6 |
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。. W/ I7 R' E% S# {& W' p
6.多加分号。+ v; A- [, n/ |
对于一个复合语句,如:
# U" ^/ [. d+ V! \% t. i{ z=x+y;
5 u- h* j7 `5 qt=z/100;) L& g8 i- \9 j5 B3 E6 A: }
printf("%f",t); b7 N% j- H! y/ L
};1 o/ i3 I( f j, ^9 n6 D2 r
复合语句的花括号后不应再加分号,否则将会画蛇添足。2 I+ z7 U$ B3 X# h% k, `
又如:1 B/ C: y1 y; a8 O1 Z& ~
if (a%3==0);
8 p! D) l1 O6 b7 @I++;4 N4 z, ?7 H$ a* C2 ]
本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
$ c4 s3 z; Q- c- W) M' b$ a再如:
$ \$ n. W/ U% V1 Mfor (I=0;I<5;I++);, A. k7 U. n6 \* H
{scanf("%d",&x);6 F9 [* X9 w, B# K6 O* U* r
printf("%d",x);}% x# I. Q- t! m
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。0 c9 F7 W4 f+ x/ _ f; o7 |: b
7.输入变量时忘记加地址运算符“&”。 a j3 o/ i& M" o" C* f% C/ N4 u
int a,b;
' B1 |* J: P0 H. Y* }scanf("%d%d",a,b);
" C$ i9 M# Z" m; P5 s, W这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。
6 X4 y4 Q* b' t- O& `8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);! {6 d, H3 O( F
输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:
- o+ N$ }9 h5 n$ \" X: M5 A6 C3,4
1 V% S$ d5 Z* F输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。
- R5 O7 T ^4 s7 D R' M②scanf("%d,%d",&a,&b);/ t: `3 b& N* C3 \1 Z# `! Z
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:0 ?! I8 E/ e' [# O# ^
3,4
1 z) y/ a) s! ^此时不用逗号而用空格或其它字符是不对的。
, W) G+ M% ^: Z3 {3 4 3:4
w9 t% ^8 k% n0 u4 l又如:
$ V5 A; ]7 c# o2 |0 h; S6 }scanf("a=%d,b=%d",&a,&b);
0 {6 `( e% c9 U* Q, @& T- O输入应如以下形式:" O) ], }" I" I
a=3,b=4
4 c2 C3 ~; ~7 H9.输入字符的格式与要求不一致。
: ~" \1 B& f2 Y w e2 v在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
2 T, v) V5 j' n. Gscanf("%c%c%c",&c1,&c2,&c3);, O* `6 ^9 C5 x$ j+ n+ o3 m# H
如输入a b c 5 i8 J7 k% d5 d5 ^0 ]
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
/ C J% u4 R O2 a. t. r10.输入输出的数据类型与所用格式说明符不一致。
( w0 [$ y; Q2 F' G& j! x例如,a已定义为整型,b定义为实型
1 r, [% ]+ }+ {a=3;b=4.5;
1 L* S/ d1 q7 q/ I" y* ~8 F, bprintf("%f%d\n",a,b);& p8 V6 K4 k. m5 {$ S
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
, J. R0 Y J+ j8 Z11.输入数据时,企图规定精度。
3 `& Q5 l2 s0 x5 e- _scanf("%7.2f",&a);
4 {% z6 i G6 |: Q6 O这样做是不合法的,输入数据时不能规定精度。; l: a. ^8 j3 O( ~7 {8 L+ _
12.switch语句中漏写break语句。
F* s1 N7 L* u& e例如:根据考试成绩的等级打印出百分制数段。
$ D R- Y2 S7 _) z+ {switch(grade)8 q$ b- t i4 w! U, t
{ case 'A':printf("85~100\n");
3 _1 D, i: ~- B; S% x' ^case 'B':printf("70~84\n");
- @8 f! J0 J& A' b+ G$ M; w8 ~0 {case 'C':printf("60~69\n");9 i# U+ {; g$ t7 x: L X9 v
case 'D':printf("<60\n");4 {& n+ E- ?0 C3 S
default:printf("error\n");
$ s! N3 Q( {3 |) g由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如6 F# B6 P; E3 ^- o+ L9 V
case 'A':printf("85~100\n");break;
8 q8 ~9 W6 p! G" E13.忽视了while和do-while语句在细节上的区别。" B0 \7 n) M: e7 y2 j% y
(1)main()
4 F$ n3 s& B' U{int a=0,I;- \3 `* j) K; F y4 p' b4 C
scanf("%d",&I);- O5 L7 z' \' Z
while(I<=10), g, H! x+ ?% ^% s# G3 `8 i$ H' V, S
{a=a+I;* c. e& b* E( q, E( k1 S, n
I++;) u2 }5 M5 k( \0 c3 A
}
, V# A" W5 d' W, Q D6 fprintf("%d",a);5 Q6 g& y$ x7 P3 }
}
" W; t3 b) c% r) @% y E(2)main(): q& U- ?" F3 Q% z1 M" V
{int a=0,I;
' K* S1 _# l% s' jscanf("%d",&I);
9 v3 w2 b$ C$ }* r/ ~6 Mdo
+ N1 L; u( G1 ^; j: G: |6 S y{a=a+I;4 l; i# ?% ~+ J7 U; R2 L
I++;
+ p0 R2 Q, s7 I7 Q& U}while(I<=10);
3 |. U9 n8 E5 M: Q" Kprintf("%d",a);
2 a$ o% T+ X! f, [5 E$ L; O. I}
( [+ P9 Z% n5 W8 J可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。
' M @9 o( a: [14.定义数组时误用变量。
7 n. E& o! H$ mint n;
( c/ S( ]( d3 W1 ]+ `% L2 L4 Bscanf("%d",&n);
$ m8 {' a+ f9 {7 s- M7 V. h w; ?7 Jint a[n];- R( V0 e5 q6 T0 t( I2 L0 M9 X4 P
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
" o. v# S( ]7 E; S" y9 B) C# c! ]0 I6 H15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。$ ?7 S2 k! \& \- k) c2 u( a
main() Q/ R9 w) B$ m4 ~* e
{static int a[10]={1,2,3,4,5,6,7,8,9,10};6 E, K8 T+ T4 ?& Z* `3 p& R
printf("%d",a[10]);6 g) g5 C. r. k* d0 z
}( u3 E$ L$ s! y' ]/ m
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。- s2 I" o0 e+ ?! z/ {
16.初始化数组时,未使用静态存储。# x. r; {/ l1 q- E, F' W! f# c5 Z
int a[3]={0,1,2};6 e( C; P$ ~: v1 ~$ l& R" S V
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:
1 u+ B& U" K# t' G' q: Vstatic int a[3]={0,1,2};4 K3 D8 n2 d* k j- G
17.在不应加地址运算符&的位置加了地址运算符。
. h* C3 I* d7 U! ~' \scanf("%s",&str);
; l0 o6 }) V$ D, H: U% h' T9 wC语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:
, j5 j" K8 F/ X1 \6 H7 x/ {scanf("%s",str);
5 F& U- a8 r6 k$ y1 |% a2 N. d18.同时定义了形参和函数中的局部变量。
$ ^; ?' }' B& l* h9 G ~int max(x,y)
4 C! d4 l5 l8 y& ?int x,y,z;
8 T5 S x/ X ~3 @% z' r{z=x>y?x:y;
( S+ \8 W: z5 k) |return(z);9 o( p) m8 n k9 `. j" I" y
}5 m1 Q& g* K- A! W0 [. n; u, ]
形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:. L* K6 g5 t3 S- K. G K$ x5 h9 J! g
int max(x,y)& G/ ~. `- q1 ~
int x,y;
! i% }$ h% o% R2 N% f{int z;: W" V% V& m( `1 P' b5 u# J/ Y
z=x>y?x:y;
1 e r, Z! f2 E0 h* [ x% ~9 Ereturn(z);
) ]7 I4 v3 ?) y% F}</P>7 a8 h/ V1 f6 T$ Q5 B1 T
< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|