- 在线时间
- 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 y7 R* q" O, `5 r$ k
1.书写标识符时,忽略了大小写字母的区别。( b1 m* H O4 b; N7 U
main()
8 }3 @+ p, `% O( A{
4 }3 [/ j9 Z, d) Y0 z0 N0 \4 \& Z0 [int a=5;+ z, C% ?5 a2 x8 i" \. r1 u
printf("%d",A);; Q% K4 Y5 t3 {9 O9 U6 i1 k
}
& E! s' K, `6 R% W. L/ I% ?5 y编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
% u, I; t. Q e1 H2.忽略了变量的类型,进行了不合法的运算。& D W G( X5 G' K: g3 F' u
main()
2 v( M/ a0 {$ x/ N$ n+ U{
4 a3 w2 n7 c# j* S6 T* ]( n! g' bfloat a,b;; u/ U2 x* L& x; k1 f2 x9 I
printf("%d",a%b);
; W& e/ i9 G' h5 w; |}
h* Y* `2 [5 e6 z6 d%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。7 S( ` i7 o7 g( r
3.将字符常量与字符串常量混淆。
( l5 C( V% J4 X& V0 T2 Cchar c;+ ~9 E& I7 r b, Q/ T
c="a"; b) J5 h3 r" Z& _
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。
: S7 a# ]+ q( O% v: J2 E4.忽略了“=”与“==”的区别。
' a) R2 K) M3 U$ B$ c在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写
+ f9 \: n* L# U1 d( G" K( Fif (a=3) then … y: R% s- |, z9 X
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
# v( D+ `. K% \5 r2 C1 Eif (a==3) a=b;8 Z1 }: x, |& C) F/ z2 U
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。
, k$ e! {" |& _" m5 k5.忘记加分号。
1 j4 B8 A* O1 `. o: v c分号是C语句中不可缺少的一部分,语句末尾必须有分号。
( ]+ O; y. L( M) v% W. J- la=15 P) N& o9 d; u5 _# u5 v
b=2
7 v2 W1 |( O7 ], u编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。
, E3 U1 L" ?9 g' D) c{ z=x+y;
0 C; w* P) V0 i) a+ Et=z/100;
j! ~- z% o+ ~8 n& i6 J. Pprintf("%f",t);
2 D- e# u, T5 x* {& i- w}) \$ j' A/ K% M/ o
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。
4 R' ^1 _0 p- F) V6.多加分号。
- m" R G" {; C. I3 E5 F) ~ J w对于一个复合语句,如:
3 M4 J r+ }8 x0 \4 q{ z=x+y;2 G; k& ^ y/ O' p
t=z/100;
o! ^) q: N' a/ D# L. Pprintf("%f",t);! m% A. |& T* h5 |
};
3 o/ J' c+ R1 o: F3 S复合语句的花括号后不应再加分号,否则将会画蛇添足。
' N4 ~1 c; _8 ?# e, p% x/ ?又如:( y1 L9 B/ q8 e: ?3 L8 \+ h
if (a%3==0);, C: s+ r" V" t9 L; P
I++;# H: C3 k. g) p: j5 M
本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
, v9 a6 M% _6 ? I# o再如:! O/ U% ? P: Z' p& d5 c
for (I=0;I<5;I++);) x3 W& {: b S( j2 X- X3 _, o
{scanf("%d",&x);/ Z7 L) M6 {9 E, o$ F
printf("%d",x);}3 \+ b( ~# G/ q& k. }' t6 u
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。
9 B! v7 `$ v% s( C4 C9 I/ y7.输入变量时忘记加地址运算符“&”。
- C( _3 q2 N. P) hint a,b;5 }3 r; l: p O( w
scanf("%d%d",a,b);, }& U3 ~# k% r% [
这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。
6 t0 [7 C3 V) o+ [. b, Y( I0 [1 R. F8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);0 ?5 I6 E" ~( K! A4 h7 m
输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:% [$ Q5 i# p4 c+ b* _+ ^
3,4
) r; i3 [8 |' b) Y5 ~! M9 n7 X5 u输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。
/ v4 G' [; G9 |( ]+ M1 ~②scanf("%d,%d",&a,&b);% [6 m1 G8 i: S" G
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:% X" L. M& [6 y% U
3,4 ; n9 h5 Y+ o( B) ^( `$ l5 W
此时不用逗号而用空格或其它字符是不对的。
! w# \1 t2 m$ f+ E6 S" J) f, {3 4 3:4 % _" w y7 ]0 Y# \, }" k2 r
又如:
5 N: z; O* D# h. hscanf("a=%d,b=%d",&a,&b);6 ^: P9 z/ p4 r; S( |- G0 ^
输入应如以下形式:
: B6 z. t* w; h4 Fa=3,b=4
8 C* X; K3 b' z7 @9.输入字符的格式与要求不一致。/ _, {: K0 f" h# P' _
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。. }$ l* }/ b+ e2 k: q; D+ D( Y
scanf("%c%c%c",&c1,&c2,&c3);' U: Y* E8 }5 E% [5 X- n) m7 o% J
如输入a b c 0 P) }) u- t* T; n2 J1 S
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。' ~0 Z, K$ g7 u1 R# r J0 {
10.输入输出的数据类型与所用格式说明符不一致。# q+ z( Q" u/ `
例如,a已定义为整型,b定义为实型
" V) O2 i' C7 A; Ba=3;b=4.5;
, H1 M0 ~8 J7 B# \2 jprintf("%f%d\n",a,b);
9 y+ u! D. Z: w2 ?7 {/ k7 b7 B编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
# [, K+ l9 Y- }! G) \11.输入数据时,企图规定精度。# r% @9 Y2 {* f2 W, ]
scanf("%7.2f",&a);" q, k x& x( N- B8 a
这样做是不合法的,输入数据时不能规定精度。
+ D! S _' E; {9 b12.switch语句中漏写break语句。
H* ~8 q, _( z例如:根据考试成绩的等级打印出百分制数段。
8 p- ]( ]% ?1 f- @) Y5 Wswitch(grade)4 q$ n4 R( ?( C w& x, U2 b" D
{ case 'A':printf("85~100\n");; b$ D, F$ \8 F7 U
case 'B':printf("70~84\n");; |+ c- a) {; V) D7 ^3 T
case 'C':printf("60~69\n");
, e r/ `0 R9 k4 ]" k5 N; t# A( ?case 'D':printf("<60\n");
% G) |0 }: {* t8 hdefault:printf("error\n");8 w: R. j% m# o8 b7 J
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如$ _* r1 m* s' x9 t
case 'A':printf("85~100\n");break;
- r9 S* A, d* b13.忽视了while和do-while语句在细节上的区别。% G+ s( |9 a# N# o! V
(1)main()
7 Z: }$ {) }- X+ `2 o w' i& \0 y/ }{int a=0,I;2 Y6 [6 L g1 T. ]' S# t( J
scanf("%d",&I);2 a4 t& f' ?' X! k
while(I<=10)
. o7 k+ Z$ W) W- ?$ K; N{a=a+I;' P4 u4 ~, t* h1 ]% a2 f7 ~
I++;
# r5 ]6 g9 t; K9 k}
3 n8 ~7 N2 }9 y' {0 t6 R/ l! s1 Y) Cprintf("%d",a);% \* ? T6 \6 M4 q o# k
}
G6 K9 p% E, q4 K9 l9 Q(2)main()4 t7 U3 t) q' T) B0 x; l
{int a=0,I;
6 R0 a( @4 E, Y, I$ n8 Z6 Z; Jscanf("%d",&I);
6 E$ v3 _9 q- e$ E* l. mdo
) [. |' C; F. k4 S: M{a=a+I;
) O! \ S0 z7 U QI++;- q* ~! L- W, T7 _1 |7 ^
}while(I<=10);
* e! D. f$ E" Rprintf("%d",a);
$ n1 {0 o6 F2 \' b/ h1 I- _5 r}
9 _* U" x- J4 M/ C w5 \4 E t可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。1 B, Y- j% }+ Y% _- k h0 _
14.定义数组时误用变量。& x) X' f5 l v8 z, V
int n;1 o( D; {) O$ J% P* l( T3 Y6 U1 m
scanf("%d",&n);: ~$ P: F1 E, [- {4 T
int a[n];
- ]1 Z- _2 B$ V$ V, ~数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。# N, W( T/ d* l/ g: k' E
15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
, X! U* w X7 V/ T4 Fmain()' u- c1 `; q. K& n
{static int a[10]={1,2,3,4,5,6,7,8,9,10};
! {- [6 X2 J1 S6 @$ }printf("%d",a[10]); \4 |9 N k3 s! B0 x/ S
}
) Z) |2 M5 M$ m- Y- FC语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。
& G8 @' d9 ? y. s* b- c16.初始化数组时,未使用静态存储。3 l0 p+ s/ X) C
int a[3]={0,1,2};
# m, I! {, A- L这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:6 G# U" q% s( L7 W
static int a[3]={0,1,2};7 Z# F7 t7 K/ x* R* ~+ u! J
17.在不应加地址运算符&的位置加了地址运算符。
8 t. `( e9 |3 Pscanf("%s",&str);
4 M6 z6 @& F4 S. c, a. I1 b4 ZC语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:
4 t- O* Y! _ t0 e0 f" c! `- g+ ~scanf("%s",str);
9 @5 U" R4 Y" y j2 x* `18.同时定义了形参和函数中的局部变量。# Z! ]0 ]* }0 a5 f% H
int max(x,y)8 Y; q. w, \0 V% M+ `& `
int x,y,z;) A( Y. @) c+ N _ x+ r
{z=x>y?x:y;
- g* `: \" i) D x% p% ^ wreturn(z);1 G- ^ k' o5 n6 @) ]1 k q' a* @1 b
}
' _1 Z3 o2 a7 y4 k* E& n形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:, P' E8 _! P3 a! C. w9 W+ D# l% _
int max(x,y)
/ X8 H6 e& V) e2 B Wint x,y;
/ T9 Y2 C4 i6 B8 C" _" Z& n{int z;
( {5 d& h# Y: O4 R: f2 B7 ?z=x>y?x:y;2 ^7 l# ^7 ?8 k4 v' V" i
return(z);7 H/ O+ T0 g& T* J! Q
}</P>2 J0 T9 T" g8 [
< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|