- 在线时间
- 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编程时常犯的错误,写给各位学员以供参考。
+ W8 r% e) d9 Y: Z! t1.书写标识符时,忽略了大小写字母的区别。9 f. C2 U. Z7 J! i1 f
main()
! P) L$ q' D* V) m7 H$ ]+ k/ F{
& c* O2 @9 Z! ]/ \5 Vint a=5;8 J7 ]& N% C4 x4 v8 H) M& ?* J l& [2 v
printf("%d",A);* L8 y( z8 F; g: n! i
}
8 ]8 Y: U% ~& x `% e9 q编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
2 b7 U _$ Q% f5 k7 j p2.忽略了变量的类型,进行了不合法的运算。
% g0 L. P3 a/ }; p) ^" Lmain()' y# @0 O. i) T7 @9 s; I. G
{
$ L/ G* `. V( Ffloat a,b;6 p- g1 `1 V: z
printf("%d",a%b);$ R" b, a' m/ u8 x9 z8 g/ q8 }* ]
}
D+ f2 g8 ~8 r- ~' r- |: Y%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。& Z3 _8 l5 ~$ h$ f5 j
3.将字符常量与字符串常量混淆。
1 a4 B9 N7 s' H7 `- ychar c;# A: A& g& q& r( o6 w
c="a";
) x1 j9 B2 W5 {# m. m7 V在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。3 `/ c9 `) L" c: j' c* Y; ~
4.忽略了“=”与“==”的区别。: `- R6 g/ S8 l1 _7 F; o
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写! v" d) _5 i) |. w) `* b$ Q
if (a=3) then …0 g& [( b- k5 Y# U' }
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
- Y) x$ a n; |$ E0 Bif (a==3) a=b;5 f: A: L5 t' \+ B8 p8 ^/ R
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。7 Z& V: b/ p6 D% F1 ~$ l
5.忘记加分号。
2 ~% ^/ w L7 r, C$ S) D分号是C语句中不可缺少的一部分,语句末尾必须有分号。2 L! B( i" K- `; T$ v
a=1
; Z. ]7 `, |, ]1 pb=2- Q" Z. ^! J9 y3 Y1 e/ g! t
编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。3 t. e2 I1 [; X- }1 f6 {" Y( Y
{ z=x+y;
& V: D* i& Z! A. \$ m# H* yt=z/100;5 A. N: g, L& c% q! D9 r
printf("%f",t);$ T; R( U L, R$ r# x3 D
}: ~! c% C0 ^- _/ c0 S8 ^
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。
( h" h, G$ o' U- v6.多加分号。
9 T" C8 a$ ^7 B. s5 H8 F对于一个复合语句,如: o% \8 ?& G* d
{ z=x+y;3 P' U( r- c$ ?# p' _# P8 i
t=z/100;
' x0 e2 U4 X# \8 ?- Jprintf("%f",t);
9 L$ v; ^4 W t};
7 N& j' B+ F4 R, N" @复合语句的花括号后不应再加分号,否则将会画蛇添足。4 `& a! P- j n9 [+ v
又如:* w! y1 N8 E5 g6 _" r( F
if (a%3==0);
$ E2 x, a( l$ |" \I++;
4 B; {1 R- u. ~; |9 P. H本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
$ _0 V# p3 i7 B# N# p& ^9 R再如:2 ]/ u- c) i4 K2 [: ^
for (I=0;I<5;I++);
" h8 {2 ~$ ]# e+ E$ t/ g# e{scanf("%d",&x);6 S' W7 m" }6 u/ ~
printf("%d",x);}' D4 G4 e1 S& O3 J0 q2 R
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。6 ^* N, p1 o# a, A g
7.输入变量时忘记加地址运算符“&”。( b4 l" q1 Q8 ]9 w- m! J
int a,b;
- \! K$ C- b; u$ Lscanf("%d%d",a,b);8 R1 _9 S/ ^4 E2 j* b' K! n$ E3 x
这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。
/ K/ U9 P) x( h2 q# K( G# O8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);
8 l' n# g6 C. g: `3 x! U: C输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:1 F* C7 P& J. W
3,4
- E8 v3 u& O6 X1 L: `# a输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。 u( N. \0 c! e6 H4 b# \
②scanf("%d,%d",&a,&b);
4 \# w+ j2 p& Y/ v' ?7 U! E8 dC规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:
* q8 y: l2 E* H! i) w; p3,4 / j+ s$ X7 q6 S( O/ ?+ q; M
此时不用逗号而用空格或其它字符是不对的。+ J( e2 q9 e( ?( @2 a9 [/ X
3 4 3:4
9 z0 }& o3 n3 D1 k又如:- v( ]+ q+ Z( G- R
scanf("a=%d,b=%d",&a,&b);, o9 w4 u& r! d0 m' p
输入应如以下形式:
- J z/ Z; G" r! R; q' _a=3,b=4
l+ u: K9 N+ K) F9.输入字符的格式与要求不一致。
0 j. d2 q5 Z% c4 X9 G在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。0 r% o0 s+ R- k" ?5 c" K: k" [& Z1 q
scanf("%c%c%c",&c1,&c2,&c3);$ \4 m8 d8 ~, d3 }- I: p
如输入a b c
. P: N0 T/ S3 U/ `5 f" y字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
2 }5 ^; W3 Q% o10.输入输出的数据类型与所用格式说明符不一致。
9 N3 V# B% S- [9 U* g$ Y9 Q" s例如,a已定义为整型,b定义为实型
1 a, p6 z9 J: p; {) M7 ~a=3;b=4.5;- ^: g) k8 r& {: R P$ u( [
printf("%f%d\n",a,b);
/ M4 v$ M" o$ a编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
: z$ y- n% }7 }4 ?. P, \8 p5 S11.输入数据时,企图规定精度。. H6 A" b* i* N; R/ a
scanf("%7.2f",&a);8 v8 `; T0 e9 |: r- c
这样做是不合法的,输入数据时不能规定精度。
+ I. `8 V1 k( k, o9 U# h12.switch语句中漏写break语句。
! Z. @8 p4 o4 e# G例如:根据考试成绩的等级打印出百分制数段。
* H* F, m- b# L4 p9 p6 bswitch(grade)
5 e `) |! u& Y& O! \ v{ case 'A':printf("85~100\n");
$ w8 C, v7 c# P* `3 Rcase 'B':printf("70~84\n");
, a1 @4 r9 }! Rcase 'C':printf("60~69\n");
2 N( v! {4 ? y9 B; Vcase 'D':printf("<60\n");
9 U1 |. p% m/ s0 Idefault:printf("error\n");
# y4 g6 Y4 Z1 s2 p由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如
* m5 E4 J! ?, [1 ecase 'A':printf("85~100\n");break;. P) i, ^* a$ `) r! L( T
13.忽视了while和do-while语句在细节上的区别。 G, o3 q! y# p9 v ?4 J0 z$ }
(1)main()
8 G) M0 b" g; N# }1 w$ F{int a=0,I;. F4 {6 B& H9 n
scanf("%d",&I);
& D* U* n# c' `. vwhile(I<=10)% r2 F G/ `& _" {; q" l
{a=a+I;
6 b, n" ?+ T# S* O9 `" Z rI++;
* K) ^) L. m- U g}7 D2 R% L1 x! {* B( t
printf("%d",a);
' {8 \3 L/ r6 f8 ?$ B3 A}1 G9 v; k( A3 }
(2)main()! m3 |7 P7 T% g6 v
{int a=0,I;
+ P# Y0 w3 Q- ? w. tscanf("%d",&I);
) c! A: b9 K* I s, e& U4 }6 ldo
/ E0 X+ x- Y7 X# u{a=a+I;
- Q, @7 w3 E% C2 AI++;' O2 o: f& q+ v' q, e0 w9 a8 l1 ^, x
}while(I<=10);
. w! t# [, I4 Z0 `' Yprintf("%d",a);
# N7 G; q7 a- C) v, _. o}
/ J) `3 e' a) t5 r6 B可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。
L' k2 }- K \: V14.定义数组时误用变量。
, T$ i! W4 g: `int n;
3 i3 ], u2 n% ascanf("%d",&n);
: v( q! e1 R- A4 s' e0 A* r, I0 {int a[n];
6 U A2 Y; d0 I( A4 p$ E# _/ t& m) I8 |数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
# `: b$ C; P4 ?15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。, ]: `/ v4 S6 J' o, o
main()* b% I, R, j! \0 a7 g( G
{static int a[10]={1,2,3,4,5,6,7,8,9,10};
0 y& J4 l! d* H. D5 v1 a5 yprintf("%d",a[10]);
* |# P/ c% n. @}5 f" V' ]; M O$ x# y
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。3 {( T A, ~ J( u3 _
16.初始化数组时,未使用静态存储。
3 w# d+ f' A6 F% N0 N* D2 W/ Mint a[3]={0,1,2}; H+ _3 c2 |$ N0 c4 z0 }
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:
6 U6 h% Q) Q& x4 Sstatic int a[3]={0,1,2};
8 b! Q7 y7 l9 q- ~7 n/ Z o1 N$ P17.在不应加地址运算符&的位置加了地址运算符。' J" v7 h. w f. f2 O' w7 a; Z% j$ r
scanf("%s",&str);( g% K H! r4 D! ]1 n
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:& T( n. A6 v1 Z6 z/ w2 a
scanf("%s",str);, `% p, t ~& Q- v: ?2 w, T4 X
18.同时定义了形参和函数中的局部变量。
6 [3 O! F: c! Vint max(x,y)
- g4 S( b3 }7 D1 c yint x,y,z;
. F9 r$ h3 R: e5 ?6 m{z=x>y?x:y;
6 I& j, Z. ~) W0 n- greturn(z);
3 N5 _8 D! A4 O5 k) X* L}
+ L2 d: t& ]: m1 q形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
* c! R" H" G! C4 f [& k' O' Dint max(x,y)( q$ M/ r5 \- [: A9 K" z
int x,y;
, q. J' k3 q0 ]2 w" J8 v" n& v{int z;
: Z3 I6 V- J3 k' A4 r+ y+ B; p% rz=x>y?x:y;
3 c5 x3 O( E( Oreturn(z);
7 f5 W+ n X2 K4 a t}</P>, d/ R" D( y' O) ]% N$ p# O; f1 O5 _2 _
< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|