- 在线时间
- 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编程时常犯的错误,写给各位学员以供参考。, z6 d* m7 C2 w! m; v
1.书写标识符时,忽略了大小写字母的区别。( P0 T' K7 {" p; a! b
main()
2 ]' ~$ o1 Y& L- ]: a6 I% ~{/ w! Q* R- y' l; |% F% _
int a=5;* ]4 i1 O2 Q3 {1 Y- \3 S
printf("%d",A);6 N$ E) F! e7 t
}
3 C1 P- f8 b- t2 |. [# m. o编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。# j9 J. g+ C; @$ F0 m1 a' ^# ]* H
2.忽略了变量的类型,进行了不合法的运算。
3 r/ L8 e9 c5 n" L3 x+ `2 M4 tmain()& q# P$ c# F+ Z! o1 S: ~
{7 |1 G$ Z6 ?. z( G, B% ?
float a,b;" l" c ^ m" q& F( N, p* X
printf("%d",a%b);
+ x+ W; R2 m) M( f% I}& \9 T2 S3 W: F, L6 _
%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。
% f: C3 J) C K( i6 r8 z3.将字符常量与字符串常量混淆。
& U' }3 f e& i: Jchar c;* F( y& s" w3 [8 N) v8 ?
c="a";, {3 i4 }6 s$ p9 u( K4 I( a
在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。- f1 c; `5 Y% z
4.忽略了“=”与“==”的区别。
1 _, t: c9 S) U, l3 D, }: Q* W在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写/ T! A3 |6 m+ z' z6 U* S( N
if (a=3) then …( h X+ D9 f% ~ r
但C语言中,“=”是赋值运算符,“==”是关系运算符。如:0 |4 _$ }3 K) [+ d
if (a==3) a=b;
; _7 p! i T; ~1 ]前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。
) s7 m4 B% v. ?7 {, N" X5.忘记加分号。9 O1 N0 O( u6 J5 b6 d
分号是C语句中不可缺少的一部分,语句末尾必须有分号。; F7 }( R( C: W
a=1
3 n7 C5 p. a: p: [% W/ {9 H0 q; Nb=2
; T, M2 x% Z- _4 p3 F% f' L编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。4 {! I+ k! y( K# H
{ z=x+y;/ v) z9 O) G& L' s n
t=z/100;# L1 f7 K5 l/ C8 S
printf("%f",t);
& Y- m8 N0 |. R9 F}
/ B0 j9 V& Y* ?8 q: @+ B对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。
5 A, @8 V$ Q8 v& x6.多加分号。& b" A/ \) j* v- t
对于一个复合语句,如:
' L) k& A7 c9 U{ z=x+y;0 o! G$ E5 ~$ G8 c- T
t=z/100;
% x) y7 t9 I" pprintf("%f",t);
, y% |% _% s! Q7 l5 p2 S};
0 J4 E! Z: x ?* W: v" ]. T复合语句的花括号后不应再加分号,否则将会画蛇添足。
; w5 y& }' G f" t4 Z5 b& Y又如:% C7 d* r/ M0 Y4 T, _
if (a%3==0);; K4 l* h+ @4 N- J( J' {$ l
I++;
' x9 l; ~6 j7 X+ h3 f% K本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。
& F! Y! D- a6 x+ T& G% M再如:
9 a% u, \; x2 @4 \' y8 D6 Vfor (I=0;I<5;I++);" p F8 h* L7 H j
{scanf("%d",&x);2 m# X! L2 ~: Q; Y5 Z) m( k7 D8 N8 a
printf("%d",x);}% y9 r6 o3 p) \8 i' h
本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。& x* k1 Q: k& s; k5 v
7.输入变量时忘记加地址运算符“&”。
6 M j3 Q1 _( M1 x6 fint a,b;! Q0 g3 [8 }+ o$ ?" ^9 o) b
scanf("%d%d",a,b);$ o! p0 Z9 p; v7 a+ ?" K: T( m
这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&a”指a在内存中的地址。. M1 ^2 a' A/ N
8.输入数据的方式与要求不符。①scanf("%d%d",&a,&b);
. B, u4 K: f" ~' z输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:( ?+ A" ]( H3 m
3,4 1 p- u. n1 N# T A
输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。
y$ j% `/ G4 v* d5 G' Z- l8 d9 U②scanf("%d,%d",&a,&b);/ I4 }2 G3 V$ u/ G [
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:
9 R1 ~) t* o6 X7 ~$ E) r! l# y3,4 4 {- @- T- C7 g' O
此时不用逗号而用空格或其它字符是不对的。" {, j4 B+ x$ [" k# `- `+ ]# p
3 4 3:4 % b" P! N Q5 [8 @
又如:+ p) h" y; b& h4 G" B' Y
scanf("a=%d,b=%d",&a,&b);7 r6 h* W7 M H1 b2 o* D1 z# z
输入应如以下形式:
& `3 k$ |1 l" y' ~/ ~7 I$ h+ aa=3,b=4 ) L9 g: K3 j. k2 {
9.输入字符的格式与要求不一致。
$ W" ~# J/ m+ R+ R+ U! l( A在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。
B$ J# Y; t+ W4 Bscanf("%c%c%c",&c1,&c2,&c3);
8 o# g2 m( Y" [7 k2 M8 H. R如输入a b c - ]8 h$ r$ F9 n7 e% \
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。0 H# R* O4 {/ @3 g$ r; e
10.输入输出的数据类型与所用格式说明符不一致。4 d/ ? ~( l5 `' r6 L" U' C+ N
例如,a已定义为整型,b定义为实型
! h: F4 J; h ta=3;b=4.5;
& m: E, N; b9 |1 tprintf("%f%d\n",a,b);
) {& a' Z8 Z. C p) p" {编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
# g) j$ Z2 l& Y$ t; N) L, z; M11.输入数据时,企图规定精度。
$ j9 Z% U- f% n3 cscanf("%7.2f",&a);
: h1 S- G; E2 C3 L7 `' j* X; I这样做是不合法的,输入数据时不能规定精度。) u" n) ?5 Q/ q6 W
12.switch语句中漏写break语句。0 Q1 j9 v* I! h7 P" Q! C) m1 w
例如:根据考试成绩的等级打印出百分制数段。
3 S% c. l; t" J7 I; u3 J9 Tswitch(grade)3 g. B& v: J0 l: q6 p7 P
{ case 'A':printf("85~100\n");+ u& `6 H C0 D# B5 q9 M% x( J1 x
case 'B':printf("70~84\n");
0 Q; u2 j8 b/ g s4 C1 J$ z( }case 'C':printf("60~69\n");
% i4 ^4 B% k+ q0 b9 C8 Q( w# }$ ]+ ycase 'D':printf("<60\n");, ^0 g+ J, _, H" o/ h
default:printf("error\n");
. e/ t2 s h6 h1 l$ d+ C由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如8 | X& y, u6 s F2 P, N
case 'A':printf("85~100\n");break;4 `6 {' l$ r% |9 I
13.忽视了while和do-while语句在细节上的区别。
6 b# O0 N2 a$ @4 {, z( R/ ?( r; d(1)main()5 O( I9 E- @/ b! h& j0 l( [
{int a=0,I;# n& g- D$ b9 p9 \& Y
scanf("%d",&I);
5 a0 b/ z$ y3 Z1 ]: Hwhile(I<=10)
) U8 [" ^, b) F{a=a+I;
0 S2 d2 w- I' kI++;7 d, S' Y2 `% O1 z
}
( m: T0 S/ Z! @# F. e' `5 v; ?printf("%d",a);4 D k' }+ }% D E' ]5 x/ L" q
}
5 x% w- { ~! A6 e(2)main()
' @/ I( N' ~ c2 i( I; t" F( N- p{int a=0,I;
$ s# _ B; Y3 S/ y# @4 yscanf("%d",&I);: W/ Z6 G- S/ N4 z C$ ~
do
0 b+ ?# F. V% y* @# @) e# G{a=a+I;6 f6 C/ y0 d, X# w9 \$ V- S1 n
I++;$ O( Y7 w( V5 v* g2 ]9 x- r/ X
}while(I<=10);7 x3 v R6 J- E8 Z
printf("%d",a);
I8 e0 b+ _" T8 F}
+ U. T7 D! q+ J可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I>10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。
+ e0 D- p* b0 m# T14.定义数组时误用变量。( Z7 Q- v3 ?9 ~* G' n+ }
int n;
+ ^5 p& T' X8 l$ }# Lscanf("%d",&n);
3 ?$ z0 V4 b! [4 c+ Y2 C3 }0 ^int a[n];
% L. b/ w+ W" P数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。* L6 z0 x- C" X
15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
/ V. M+ B, l/ U+ N- O- Kmain()
. V* ]& K; m( p3 i{static int a[10]={1,2,3,4,5,6,7,8,9,10};
4 J2 J5 G" L9 }$ H" d1 Aprintf("%d",a[10]);
9 g* T. `8 S$ D9 J F$ y+ {( U}3 q* N" g+ N& }1 k D2 I
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。
* N3 m7 k' n0 z0 S* L, G16.初始化数组时,未使用静态存储。. @0 B" q' Q( E( n* B. G
int a[3]={0,1,2};/ Z0 r1 _9 o, y, }
这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:' G3 X& R! _6 |2 t6 C/ t Y, i
static int a[3]={0,1,2};
/ ]6 C- Q- }$ L7 [5 e4 x17.在不应加地址运算符&的位置加了地址运算符。7 Y; V9 _- C+ |( z4 }
scanf("%s",&str);
: T& f# T( i5 w0 `, oC语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&。应改为:
1 N( O0 y2 ~# {$ Tscanf("%s",str);8 x7 {2 F1 ?8 k0 z( M! J( f$ M
18.同时定义了形参和函数中的局部变量。* W. j8 L- C% K; X4 H
int max(x,y)
4 ^5 g) A4 i- h( y/ lint x,y,z;3 |9 T3 K# p8 X, c$ H. P* X
{z=x>y?x:y;
2 @$ r% w3 S# |4 Rreturn(z);
- k) {! A F: Y- p}
1 Z( t5 L- x5 I P- a# N: \形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
* J1 s9 J+ O. m1 iint max(x,y)8 H' V1 `6 \: B1 c
int x,y;
* V3 k- l( G0 w9 M n( V{int z;4 z9 t9 \) F) x
z=x>y?x:y;
5 z c. ^) {* Q: U+ Z- ^& treturn(z);
6 u) l/ B, ^4 ]# c, g}</P>
* h- U8 {& T7 }1 @. X7 R8 e+ a- m< >以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P> |
zan
|