- 在线时间
- 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 _! R$ U# n( w3 t+ ]
1.书写标识符时,忽略了大小写字母的区别。 w# U& i( |# x( V
main(). \1 u1 o. d: F o" A7 Q; H
{
0 j: T$ x: E; u# w7 `+ @( bint a=5;
4 r( \' ]6 M9 V1 n9 x% g/ ~printf("%d",A);% p1 L2 F+ c' J4 D
}5 M. b# z* c( H, ?$ G
编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。
- k: G8 s: h4 H( @8 @! {2.忽略了变量的类型,进行了不合法的运算。
+ ?: v1 v* L# J+ jmain()3 \' G. ]: |) D2 ?- G
{
3 P1 W2 v3 O" Q4 Q! {+ gfloat a,b;. x I& g5 q! Y1 M( L0 b, o- W( r
printf("%d",a%b);; g* ]& f& S: C. {, `/ l
}" c v6 U" n7 t" w5 u% I
%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。
* ?" f/ Z5 z9 ~# ~! F$ h3.将字符常量与字符串常量混淆。
* }4 c" {! X7 ]9 M. Kchar c;
7 k3 m: [( S) H& a+ N& Fc="a";6 x7 l" C8 a8 [* M; U0 ~7 ?" y2 s) K
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。6 w1 M, ~4 I2 P- c
4.忽略了“=”与“==”的区别。; ^9 v- \. |1 ~% E1 s' n" i; O
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写
" u' u9 l9 M0 S# Q% K8 @if (a=3) then …
2 `4 k/ L+ i8 q/ w但C语言中,“=”是赋值运算符,“==”是关系运算符。如:* h- q C/ x7 G, \% o ~; ~9 K; n
if (a==3) a=b;9 h3 X) g- V: t( {- w8 h0 S
前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。! W1 f- `1 G( I+ D# Y& O9 C0 s
5.忘记加分号。- s& B$ L* u8 s1 K4 m" _, m; l; H
分号是C语句中不可缺少的一部分,语句末尾必须有分号。
: ?* c8 M6 H. g$ F Ja=1
: R" N! l5 _* Y. u( Yb=2
: |* \7 y8 T3 W0 l' _编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。' D$ M, f Q# w8 o( }: Q
{ z=x+y;
0 L9 m7 V* Y4 {t=z/100;( G9 E" `5 G8 r; U6 G
printf("%f",t);
I, \6 \) q2 g9 @1 X4 v5 |/ d} E. r- ^& h' O- U: v) y% y6 D4 B" o
对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。
# h9 p$ J4 Y( ~! I! Z% K6.多加分号。' J8 b1 a# [2 g6 s5 c0 c
对于一个复合语句,如:8 d, y' b) m: O' K
{ z=x+y;
6 x1 b% f/ w5 ^ @ o* yt=z/100;4 B! R8 ?# L0 I
printf("%f",t);7 h1 q* x# l6 D8 ]2 T5 T( `
};. X7 @) {( j4 }' n! M2 O- W
复合语句的花括号后不应再加分号,否则将会画蛇添足。
- e4 U4 m1 ^! \+ @6 X又如:0 n7 O1 O2 i* f% @. u9 r# F
if (a%3==0);+ Z; V% \' U8 i( j* v! T `, z
I++;
: S- T5 B, Y0 q4 E2 U本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。" P, m) Y/ C6 @) G' W
再如:
, w S2 X! Z; g2 ]for (I=0;I<5;I++);
! [; [, n E1 K( Z' \) M: Q, z{scanf("%d",&x);1 x+ N' h3 K! M1 b; w; |
printf("%d",x);}
4 S" h) W3 j% _9 X5 p' e- S+ a本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。4 q& _% Q7 U; [" c. u
7.输入变量时忘记加地址运算符“&”。9 Q* w9 @( C4 f$ W; P
int a,b;4 o4 [! i/ x5 F5 J. k. H+ E6 A
scanf("%d%d",a,b);
) z& o! p2 O- V y7 U' U1 t这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。
& m1 q4 N7 p. B3 i9 ]' `8 Y1 T8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);+ Q' _9 |( f8 v- M2 u) w' L/ ?
输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:: H/ W3 V& P8 x3 i0 v
3,4 ) F/ D& T# F3 C8 Z$ @
输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。
) X8 j. b3 @% L1 T* B②scanf("%d,%d",&a,&b);$ P4 R7 u K! d
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:
# x( ~* w Z, n2 U3,4 8 Z, ~$ g. I" K' [+ `& U& [
此时不用逗号而用空格或其它字符是不对的。
# o) r- j2 G2 }3 \8 G# _" G3 4 3:4
$ a7 S T7 U$ n9 c又如:
! J) u6 W' {/ X, ]- `scanf("a=%d,b=%d",&a,&b);
x8 P7 R! O- ]+ X输入应如以下形式:
# u; U7 D- n, e- j, Ha=3,b=4
) B% C y) l, v1 r9.输入字符的格式与要求不一致。. l9 d$ E j# P) K3 W
在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
4 ~2 e2 E4 B4 a( M' ^scanf("%c%c%c",&c1,&c2,&c3);
/ j+ Z- w* J3 S如输入a b c ; `; f: {- w+ J( p4 N
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。' ~. p8 y& D/ L, @
10.输入输出的数据类型与所用格式说明符不一致。
% W( b; D4 d3 A8 C, I6 [例如,a已定义为整型,b定义为实型2 g# c- Y" k) A7 V6 J; A4 \2 k& S
a=3;b=4.5;7 C4 n1 h% r6 `% }# X Q6 Y
printf("%f%d\n",a,b);; G" c4 E$ u4 U, E+ \# l
编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。! M3 g: K$ [# z8 X# T- F4 V
11.输入数据时,企图规定精度。8 a. e* _. z u. J; v+ j! S. ]; `
scanf("%7.2f",&a);
3 a6 J( w. A- k5 h2 j3 N! H这样做是不合法的,输入数据时不能规定精度。$ e) W) `+ T+ V+ X: m. [
12.switch语句中漏写break语句。
$ p4 K6 V+ O9 i: }+ o/ A! G例如:根据考试成绩的等级打印出百分制数段。
% U( e3 Z8 f1 M" ]1 Y! Yswitch(grade)4 S$ C7 b) [% D1 q f8 h; z: [
{ case 'A':printf("85~100\n");
$ y" y# X, j1 ~8 x% jcase 'B':printf("70~84\n");
. N4 Q( E1 k- F( E; n& B) dcase 'C':printf("60~69\n");
( m3 m- L6 F5 j- @+ s/ Qcase 'D':printf("<60\n");- |* j4 @3 p$ Q7 S2 o
default:printf("error\n");* Z+ U; _* _& e$ f
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如/ I7 [% N& R" Y# p. X
case 'A':printf("85~100\n");break;+ u! x7 a- _* r8 N
13.忽视了while和do-while语句在细节上的区别。4 b3 A; [; Q3 s7 R# `- A, @
(1)main()
, v3 }3 W/ _! S: O4 s: H, l{int a=0,I;
+ s( ^/ |/ ~; _8 a5 B) R; T; ~1 cscanf("%d",&I);
; s a i0 E- D0 l$ S2 Awhile(I<=10)% \- K( m. g! H% X
{a=a+I;2 F7 ?2 G9 G0 C, Q
I++;; f$ u! o) c5 s- E6 T1 m0 e
}% a- Y3 L: T- \4 i/ z" H1 B
printf("%d",a);
s. Y* c% W' b, [ s}! D; z1 P/ C, R9 b- V5 |, U
(2)main()5 m0 ~ k" g, q8 b( H
{int a=0,I;4 l2 r3 ~0 Y4 V& @# L1 A6 \
scanf("%d",&I);7 N- N7 r! w# I. N6 l/ p
do6 E7 x; J" c1 y% g
{a=a+I;
! ] P1 m9 h" \/ m5 }, H8 dI++;
. o8 g6 V" C3 \% I7 p: r' _}while(I<=10);
! {: U9 b' g% Q. P/ |; Iprintf("%d",a);$ y4 |" M+ ~9 ?
}' n0 H/ [5 c! h4 o+ d
可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。( U. J; ~/ F% O2 f: j. N
14.定义数组时误用变量。* }) ~; f( G! l2 O* o
int n;: L) X4 S7 z' V/ F" A* i
scanf("%d",&n);
, ~3 _2 w V7 M# bint a[n];! _: N/ `: s7 d6 I
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。
1 w! F( A0 f! Z4 n" m: F6 U9 `15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。+ V3 D7 W7 O* U% k
main()
2 X" d! |' C; K! N- p% Y: g: e{static int a[10]={1,2,3,4,5,6,7,8,9,10};4 Q3 u! A3 @9 L8 \# Z! q5 ~: _9 q
printf("%d",a[10]);
7 Z$ s9 U1 f/ ^}
m# m- ?. e+ Z2 t" l9 b' sC语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。6 R, b% X1 v, U
16.初始化数组时,未使用静态存储。, D1 S" }+ `* X# y( g* Q7 i
int a[3]={0,1,2};: v) q6 b+ E0 c( }! w, A: N
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:& C0 Q `& X4 D7 Q/ D8 x7 K- |- X4 z8 M
static int a[3]={0,1,2};9 R3 G$ E: W( P) g# ~
17.在不应加地址运算符&的位置加了地址运算符。8 P0 @2 |5 I+ K8 L1 l& U
scanf("%s",&str);2 W1 {* b* x2 n: b
C语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:
: | a7 e8 j* D, K8 ~% w& e: k% cscanf("%s",str);1 T. S7 D' z: H! Q3 v4 C0 G
18.同时定义了形参和函数中的局部变量。0 F* w0 a1 ^" [; z; z% D4 w4 p# D
int max(x,y)9 m8 [4 t1 i& E9 \& ^# Y1 O
int x,y,z;
( B9 ~% X) B, Z% F1 k* S5 d9 s{z=x>y?x:y;7 s8 [% _/ X5 T% F
return(z);9 L# G k8 Y3 h! G! g4 U9 X
}
% e: Y; t$ S2 r6 }9 n f形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
0 ~! c5 F) w8 a8 sint max(x,y)( c' f1 s& Z F. B- G
int x,y;
: W4 f% e' a' z0 a A{int z; G$ K+ {" k0 d* q4 R
z=x>y?x:y;
& K$ W; z n* _$ @6 yreturn(z);: Q3 ]% d( u+ z" y
}</P>
5 m; h, Z& H- \9 a; F N4 Z! U< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|