数学建模社区-数学中国
标题:
[转帖]++C语言中常见错误++
[打印本页]
作者:
solucky
时间:
2004-6-6 19:01
标题:
[转帖]++C语言中常见错误++
<
>C语言的最大特点是:功能强、使用方便灵活。C编译的程序对语法检查并不象其它高级语言那么严格,这就给编程人员留下“灵活的余地”,但还是由于这个灵活给程序的调试带来了许多不便,尤其对初学C语言的人来说,经常会出一些连自己都不知道错在哪里的错误。看着有错的程序,不知该如何改起,本人通过对C的学习,积累了一些C编程时常犯的错误,写给各位学员以供参考。
c/ A& |$ x' ?4 f
1.书写标识符时,忽略了大小写字母的区别。
# C2 _. L4 S# z! n# u& Q
main()
/ U' C. p( W. D( D) V9 ~
{
, v& O' M5 U. l
int a=5;
1 {5 @6 d$ a: h6 r7 I J
printf("%d",A);
# W5 }1 _. I. ] H8 o& [; m m
}
. B3 o4 b4 w+ F3 H9 A7 W6 a5 N1 G
编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
1 U3 c! Z6 r$ k L/ j- k: ~
2.忽略了变量的类型,进行了不合法的运算。
0 s9 o: M/ P1 G
main()
9 f7 H' A. }- w" b/ B5 n
{
1 @% |4 |- ^& x
float a,b;
6 v7 b, x4 I; m3 [
printf("%d",a%b);
" V" ?) N. N5 R2 ]
}
5 V& x F) w# k' P! ~; T
%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。
2 Q5 t+ r" M4 F# H3 E' W) }
3.将字符常量与字符串常量混淆。
( @) Y. G0 @( ^7 ~9 n# o
char c;
0 _$ I8 k5 X/ w
c="a";
6 k: T# k) K# B4 k
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。
, z/ W4 V2 K" ~
4.忽略了“=”与“==”的区别。
8 S) |2 z* z4 {+ e; w& ^
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写
. X! q7 ]& b: x6 h# i: k
if (a=3) then …
) v2 k' b; ^( s' k
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
* v* ^) ~8 ], x, U, q7 J
if (a==3) a=b;
$ B' G8 C, _+ D8 E& D8 G9 I$ N" ] D
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。
0 R0 [: }# p9 H
5.忘记加分号。
7 o* |+ N/ o; h3 `
分号是C语句中不可缺少的一部分,语句末尾必须有分号。
$ v( E! }7 `+ G
a=1
X( G: ]4 Z, A* O- v G" Q e
b=2
' z! G* u. N0 m# m
编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
9 Q( ^+ k3 O$ V7 h" m$ W
{ z=x+y;
! F. t D. b [# _7 P+ R3 r8 M
t=z/100;
3 j5 A& I8 `. n; D
printf("%f",t);
; k9 s" b1 w8 L x* o
}
/ S9 B5 E0 F) \# z% t! l# h
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。
) P2 u8 [" f$ U- V" C1 }
6.多加分号。
. B# |5 B' l' h/ _& \
对于一个复合语句,如:
, W: Q( }* i' d0 g7 _+ {
{ z=x+y;
6 B8 v; _+ `$ V, V) W- X$ w4 H& P( ~
t=z/100;
& g0 X. V# \, }! g$ Z9 {* m
printf("%f",t);
5 w2 K' Z% ^8 O4 U$ s' k" h
};
! @2 H5 L. S- v5 H9 \6 a* {+ [( N
复合语句的花括号后不应再加分号,否则将会画蛇添足。
0 [4 |& O- L( z6 F( o+ k
又如:
2 @5 C& P, z0 m
if (a%3==0);
: e% K: [( e" n" Y1 b- v
I++;
+ `) L4 D0 F) \* K* T! p* }! {* E
本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
: [: J, y/ r! K7 n" N
再如:
I0 v0 I% W3 @7 l' L, B
for (I=0;I<5;I++);
- ?* ?6 U" V/ l* B- ?: Z3 H
{scanf("%d",&x);
$ d+ L8 y" I# L6 p& t2 L
printf("%d",x);}
; ^- m0 ?+ [$ s! h# D2 z
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。
& `2 V5 M% l8 Q G7 r& w
7.输入变量时忘记加地址运算符“&”。
0 D) d) b: w1 W, p/ M/ w
int a,b;
4 Q# ]9 G" n' b! |+ r, K7 F* |
scanf("%d%d",a,b);
/ \- S7 `9 _5 |. D% U0 a
这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。
# c/ a7 a7 S& Q; z, t, i$ l
8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);
8 O3 z+ P* X# C. M: F R
输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:
$ N- \% A$ }- n) M: W4 _- ~
3,4
1 q% L8 \0 c: p( N5 R* D
输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。
5 G# V3 K% ]- z& i u- L& U
②scanf("%d,%d",&a,&b);
; @: t( @% z* R, M% [% v5 ~7 n& _
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:
* t- v0 J P6 T, a! E
3,4
" E+ N" I/ I4 R1 N3 k) `
此时不用逗号而用空格或其它字符是不对的。
3 f1 H3 U1 d. w4 K2 p) p0 ?
3 4 3:4
8 q% h6 R7 E: Z. x3 p1 G1 A
又如:
, h' }6 B3 c! B3 O0 m& m- D4 _
scanf("a=%d,b=%d",&a,&b);
- P* z, I7 m" |# d0 ~
输入应如以下形式:
' o2 E( E- \% l, q% M
a=3,b=4
* D& z: R$ F/ {3 H7 `1 g
9.输入字符的格式与要求不一致。
+ p2 [0 j- M3 l3 F
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
( K) k. V3 v4 k4 f% _) i/ {
scanf("%c%c%c",&c1,&c2,&c3);
! @5 K" }" [, H* z0 D/ E* A/ }* \7 }* a8 N
如输入a b c
* b& D$ O8 e9 W: l) N
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
. l6 W6 |+ W& w# O6 e8 A/ n
10.输入输出的数据类型与所用格式说明符不一致。
" ^# j( x* E2 q
例如,a已定义为整型,b定义为实型
7 ~4 w4 ]5 O- V7 ]8 T
a=3;b=4.5;
- k. H0 R Q g# h0 [( r* }
printf("%f%d\n",a,b);
, E& V/ V" a) |+ H/ l# O! Z: ~
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
% N3 j% w# {$ N" U2 H Z# a
11.输入数据时,企图规定精度。
) {5 l0 C {* d2 D6 \9 k6 \ R' F& D
scanf("%7.2f",&a);
! t& W" s4 d! ]- m2 J% M' n
这样做是不合法的,输入数据时不能规定精度。
! M) A9 M/ ?# q
12.switch语句中漏写break语句。
7 z- M* S# S, l, V+ N8 ]/ K
例如:根据考试成绩的等级打印出百分制数段。
: G, D9 I5 E- j& H; b9 t
switch(grade)
% ?5 O' h% `1 A% H+ E
{ case 'A':printf("85~100\n");
8 x$ D( q3 V6 C
case 'B':printf("70~84\n");
6 v6 f9 T3 _; h9 `
case 'C':printf("60~69\n");
- F$ O9 d* u- F+ @
case 'D':printf("<60\n");
, B" @' b% [. f6 e: Q# Y
default:printf("error\n");
+ {9 |7 @+ S0 l1 p! Z4 p/ r9 O
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如
/ |) v; Q: x6 W, x. Y
case 'A':printf("85~100\n");break;
/ l- F. A# |4 ]( f
13.忽视了while和do-while语句在细节上的区别。
: f5 W* X4 ] [: x) H2 _& _- l6 G5 T
(1)main()
8 B- X9 A- i1 d# v
{int a=0,I;
7 D! [ j, c3 y: Z/ C6 K; v
scanf("%d",&I);
5 R) F4 l$ Q& O2 d1 }2 p1 {
while(I<=10)
+ a# C& @1 m/ M. x
{a=a+I;
# d' P) ^# N: N9 A
I++;
+ c+ H. Y% n8 J6 r# @
}
3 G0 ~- y( @& w7 a# S. ?
printf("%d",a);
) [! l3 t9 H5 z& C% c+ E
}
* J& q- D2 B5 e% Q; u& D' W
(2)main()
" @: } m/ v' r4 u- r; I
{int a=0,I;
7 f7 d( F' ^- v. q
scanf("%d",&I);
' J0 P# h! V B; M) i
do
* v/ o! x* q v# e2 A
{a=a+I;
( Q& k6 E7 ]" R& y( _
I++;
; u3 L6 k3 p8 S! n5 S; Y
}while(I<=10);
, h% d# _8 F* Z! M5 z4 L
printf("%d",a);
5 F0 A) l; ~- a2 o! U
}
j! X5 A% u- h. [! r ]
可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。
2 F5 E9 U' q- J' M# D. ]
14.定义数组时误用变量。
6 ?5 w0 `( b' p$ d, H: E
int n;
w, B+ T' |& p7 ?5 T u: a
scanf("%d",&n);
. P) D+ h s" [* s
int a[n];
" _* X* w! n- T- h, t4 h
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
" U5 X! h) t4 [ E3 N" E4 M' x
15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
& C Y% `% c1 w" P5 H& U
main()
4 N! x& y% k* @/ n
{static int a[10]={1,2,3,4,5,6,7,8,9,10};
- o5 s1 x& j- F6 F# k% u5 }
printf("%d",a[10]);
2 }% a: R o# |1 Q
}
! ?) o' R. H: h) H! u$ C
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。
# D* r) P) |. z& a8 q* c- q: W% q
16.初始化数组时,未使用静态存储。
: k8 Q C" L( L. C8 a0 Q
int a[3]={0,1,2};
; `% @4 O( b8 ^4 l
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:
* Y' U, I5 A7 a2 e4 b
static int a[3]={0,1,2};
3 |4 ?6 \6 J6 x! W2 [6 `, a
17.在不应加地址运算符&的位置加了地址运算符。
7 j! k! p! v, d# P9 I
scanf("%s",&str);
7 ~- d# ?' j9 h2 _
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:
' s( v# N4 Z; r
scanf("%s",str);
8 _9 z$ U6 W3 y1 G1 l1 Z
18.同时定义了形参和函数中的局部变量。
a# k2 V7 e9 |/ y
int max(x,y)
$ j* c0 C# O' l/ z* m0 f, M
int x,y,z;
, e* D" k/ `% z( d% e. G
{z=x>y?x:y;
9 J. F$ V" f2 J. ^
return(z);
1 \4 ]9 [$ s7 v0 M# S! O! J
}
: C6 N: a; ~% e. v1 g- G& b
形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
% [* K+ {& z5 J* r: X+ {$ ~
int max(x,y)
7 U# p* t p5 M- U. n7 [, y# C
int x,y;
# {: m: g. n$ h/ i
{int z;
2 Y g5 |8 Y7 l0 F& j2 A
z=x>y?x:y;
4 m0 Y R2 _% q
return(z);
/ |: y5 E# y3 D$ Q i
}</P>
( W) p5 t* `9 h0 r) {
<
>以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P>
欢迎光临 数学建模社区-数学中国 (http://www.madio.net/)
Powered by Discuz! X2.5