QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2926|回复: 0
打印 上一主题 下一主题

[转帖]++C语言中常见错误++

[复制链接]
字体大小: 正常 放大
solucky        

17

主题

1

听众

109

积分

升级  4.5%

该用户从未签到

跳转到指定楼层
1#
发表于 2004-6-6 19:01 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
<>C语言的最大特点是:功能强、使用方便灵活。C编译的程序对语法检查并不象其它高级语言那么严格,这就给编程人员留下“灵活的余地”,但还是由于这个灵活给程序的调试带来了许多不便,尤其对初学C语言的人来说,经常会出一些连自己都不知道错在哪里的错误。看着有错的程序,不知该如何改起,本人通过对C的学习,积累了一些C编程时常犯的错误,写给各位学员以供参考。: f' Q8 |1 `9 ~: M/ I- X4 i6 U) Y
1.书写标识符时,忽略了大小写字母的区别。5 \+ e4 D4 D2 G" S+ {
main()
5 l* T) E& h; |( {/ W{
+ t% X8 O- |3 t# wint a=5;9 |8 _6 Z4 L! y& a' W
printf("%d",A);
' T) m/ g& R$ d: \% V5 X0 t- {( l9 ?}
4 T3 y% ]. ?9 f+ g7 n3 R4 j0 K+ V编译程序把a和A认为是两个不同的变量名,而显示出错信息。C认为大写字母和小写字母是两个不同的字符。习惯上,符号常量名用大写,变量名用小写表示,以增加可读性。; Y' U* j5 H, a$ M
2.忽略了变量的类型,进行了不合法的运算。
" u! y' g: r( Omain()
5 \4 E* \: E# Q, z. L$ L{
% ?0 o7 _/ K; ?+ dfloat a,b;
$ U0 |* S4 {' Dprintf("%d",a%b);
0 v3 _: r, o: I0 N0 C& W  a}
. p, @& v5 ~' \3 r5 A! y0 `9 J%是求余运算,得到a/b的整余数。整型变量a和b可以进行求余运算,而实型变量则不允许进行“求余”运算。/ p# n* V6 R7 g3 T  S
3.将字符常量与字符串常量混淆。! z8 ?- T4 F% l; x- G2 S5 P
char c;
0 q, }& E% a3 U" Nc="a";
+ u3 u; p' D. L; s. |" w在这里就混淆了字符常量与字符串常量,字符常量是由一对单引号括起来的单个字符,字符串常量是一对双引号括起来的字符序列。C规定以“\”作字符串结束标志,它是由系统自动加上的,所以字符串“a”实际上包含两个字符:‘a'和‘\',而把它赋给一个字符变量是不行的。
' d" @- O; M# _4.忽略了“=”与“==”的区别。9 ^$ w" m% I% V5 r9 S0 f7 u5 C# t
在许多高级语言中,用“=”符号作为关系运算符“等于”。如在BASIC程序中可以写, O2 V$ A6 z* ]  [& T% [5 v' b
if (a=3) then …
& r, P( E3 O# X( T! {但C语言中,“=”是赋值运算符,“==”是关系运算符。如:
4 a( o- H8 i; ^; Pif (a==3) a=b;
* S8 w: d1 h7 Z$ i前者是进行比较,a是否和3相等,后者表示如果a和3相等,把b值赋给a。由于习惯问题,初学者往往会犯这样的错误。8 i$ M$ ]. ?) G- Q
5.忘记加分号。! u) Q& ?# Z' [& p- E1 w7 A6 B+ ^
分号是C语句中不可缺少的一部分,语句末尾必须有分号。1 p- l9 `' x- J2 t# n" g; F
a=14 P5 y8 d& j& D7 H* q+ e
b=2$ B9 q: C' W9 ?# D' K7 [: }
编译时,编译程序在“a=1”后面没发现分号,就把下一行“b=2”也作为上一行语句的一部分,这就会出现语法错误。改错时,有时在被指出有错的一行中未发现错误,就需要看一下上一行是否漏掉了分号。' ?" E0 L$ _6 _4 x4 ?# Q
{ z=x+y;( t" l; i6 D. k( X! U
t=z/100;) c$ S* g0 ?& P: d; {7 ]
printf("%f",t);$ A- [; g9 H6 L+ a, D( q
}
# o$ w, \+ c5 w1 [; z+ [/ [1 d对于复合语句来说,最后一个语句中最后的分号不能忽略不写(这是和PASCAL不同的)。! }1 G; `# {, U! |; ]4 H2 n0 d
6.多加分号。* k2 u" R! y( f0 C6 Z+ M7 q( q
对于一个复合语句,如:1 j1 p7 S9 u3 s4 n
{ z=x+y;
$ A2 s2 P$ e! It=z/100;
7 H5 W6 v4 Y, r$ q! kprintf("%f",t);
" s4 d' L* ?' [5 k};) \. O- @6 _5 s/ A. n; @; y' t
复合语句的花括号后不应再加分号,否则将会画蛇添足。; c" ^) f+ W0 L. _: s, o! _
又如:' ], n! s0 u5 E5 d% H
if (a%3==0);2 C7 X  |0 `0 _: C
I++;
& n4 r- A) L" l. @+ R  O4 D本是如果3整除a,则I加1。但由于if (a%3==0)后多加了分号,则if语句到此结束,程序将执行I++语句,不论3是否整除a,I都将自动加1。9 _- `8 @- |. S. F5 _
再如:0 j2 w4 P+ n+ B) L7 ?
for (I=0;I&lt;5;I++);* n/ V5 z* t9 q) o, A1 @  z
{scanf("%d",&amp;x);5 z6 V, ~0 w$ W& G2 g8 o
printf("%d",x);}
7 {, P* {! T+ K) @- w& F本意是先后输入5个数,每输入一个数后再将它输出。由于for()后多加了一个分号,使循环体变为空语句,此时只能输入一个数并输出它。
! H! e, u5 i+ u7.输入变量时忘记加地址运算符“&amp;”。
5 K) y3 i) O$ S; c1 Uint a,b;
9 X9 |2 A" ~0 f& Q+ @scanf("%d%d",a,b);
2 }- B( A* {0 v! W0 Z& |这是不合法的。Scanf函数的作用是:按照a、b在内存的地址将a、b的值存进去。“&amp;a”指a在内存中的地址。# Q* L3 S- J8 d# R, ^3 \
8.输入数据的方式与要求不符。①scanf("%d%d",&amp;a,&amp;b);
9 Q" z& h: B" I0 s' J/ O' g输入时,不能用逗号作两个数据间的分隔符,如下面输入不合法:
$ b& p. w/ F# E0 T" v' B3,4
+ _- r' j( x8 @, T  e输入数据时,在两个数据之间以一个或多个空格间隔,也可用回车键,跳格键tab。: |1 `8 u. F% n
②scanf("%d,%d",&amp;a,&amp;b);* {+ Z0 u0 D6 |% a$ K; }/ E, S
C规定:如果在“格式控制”字符串中除了格式说明以外还有其它字符,则在输入数据时应输入与这些字符相同的字符。下面输入是合法的:; L. b6 E2 j+ v5 ]7 ]8 q# X( U
3,4 : ]6 ~# m6 A# M3 }" y1 e2 P
此时不用逗号而用空格或其它字符是不对的。
7 ~- w# h2 C% z  b3 4 3:4 4 S* ]- G& R) f1 d( |3 {0 C
又如:
9 E4 Z3 L* |$ M2 N# T# p- tscanf("a=%d,b=%d",&amp;a,&amp;b);. J7 i# U- Z/ G7 c. T, o
输入应如以下形式:
( G; r6 R4 T5 h( e3 b3 va=3,b=4 & I7 z# B; Y7 K+ d
9.输入字符的格式与要求不一致。
: j2 V/ O# i. k9 E在用“%c”格式输入字符时,“空格字符”和“转义字符”都作为有效字符输入。9 n. w$ ]& ^# L" Q' w- t; y9 @' y
scanf("%c%c%c",&amp;c1,&amp;c2,&amp;c3);
& x3 r2 f* X* k9 b% T  ?% L如输入a b c 0 L: N) ?7 y, p6 s
字符“a”送给c1,字符“ ”送给c2,字符“b”送给c3,因为%c只要求读入一个字符,后面不需要用空格作为两个字符的间隔。
( m" D$ }4 l: O8 T10.输入输出的数据类型与所用格式说明符不一致。7 e- q4 c5 L$ C
例如,a已定义为整型,b定义为实型& m3 Z# V, [9 c- F; l9 Z( \  b! L
a=3;b=4.5;
5 T6 u9 _7 n, B6 ~, U7 X. Vprintf("%f%d\n",a,b);
! @/ E) m/ @# ?1 B% n* y6 v编译时不给出出错信息,但运行结果将与原意不符。这种错误尤其需要注意。
9 k7 I+ X1 \1 p' b7 H9 r! s11.输入数据时,企图规定精度。' b- n% |/ T: m
scanf("%7.2f",&amp;a);& ~1 d) x1 z$ ~
这样做是不合法的,输入数据时不能规定精度。
+ {& e8 F. j, a6 h7 u1 W12.switch语句中漏写break语句。
" q& F) j  e8 o例如:根据考试成绩的等级打印出百分制数段。
* X- U% z' i8 k, Gswitch(grade)' i: _2 t' \# @8 ^, Q
{ case 'A':printf("85~100\n");+ O: ^# K  e# n$ L$ g/ Z
case 'B':printf("70~84\n");& n8 U" {  ], r/ e
case 'C':printf("60~69\n");# T0 r& [$ {: B/ d
case 'D':printf("&lt;60\n");. e( j. Z6 [3 E- ~) i
default:printf("error\n");; h( ^% _" l9 h0 l. s! B
由于漏写了break语句,case只起标号的作用,而不起判断作用。因此,当grade值为A时,printf函数在执行完第一个语句后接着执行第二、三、四、五个printf函数语句。正确写法应在每个分支后再加上“break;”。例如+ l8 [8 Z/ Q, D% c- E: _" E
case 'A':printf("85~100\n");break;8 ]: m3 ?+ |1 M9 G- i( ]% B
13.忽视了while和do-while语句在细节上的区别。
4 g0 e3 Z* ^" H9 \2 i& D& O2 P(1)main()
0 S& G5 l4 N: D8 H{int a=0,I;' ^5 _8 y) t+ O  L
scanf("%d",&amp;I);& @# b0 d. {! j! B; k8 P
while(I&lt;=10)
. }$ ]" Q5 w2 A" L: V: a- W{a=a+I;! b6 e2 F$ X+ E* V, @, |  k
I++;
2 f& U1 C, _3 L6 |) o}
" _5 f* {% m5 L/ dprintf("%d",a);7 k# b  i3 g/ v* i$ ]8 e5 U
}
3 w+ \1 f$ b# L" [  b- m! L(2)main()$ |5 r& B' J& {0 N: u$ k
{int a=0,I;
+ N2 |; z) Y3 b' ]. Jscanf("%d",&amp;I);% N2 p- L, D( a( f' v
do7 ~8 p; x" q8 E" D# n7 j
{a=a+I;9 A% m+ L+ ~* B
I++;3 J( V* S5 T. n7 X# c2 H
}while(I&lt;=10);( z& G( K# o: @- G) }  z9 C
printf("%d",a);
+ {* Q+ ?: I+ `, E7 }( [) w. d}
4 @+ ~4 f+ s6 c! z& u3 T可以看到,当输入I的值小于或等于10时,二者得到的结果相同。而当I&gt;10时,二者结果就不同了。因为while循环是先判断后执行,而do-while循环是先执行后判断。对于大于10的数while循环一次也不执行循环体,而do-while语句则要执行一次循环体。. j2 m* J9 Q( w% X
14.定义数组时误用变量。$ A' q4 E- I* L( H, o( J
int n;# `/ I. ^. d2 l' {
scanf("%d",&amp;n);
! k; Y! [5 _1 W6 |3 Rint a[n];% g' n$ o0 G* N$ H- u* j
数组名后用方括号括起来的是常量表达式,可以包括常量和符号常量。即C不允许对数组的大小作动态定义。* k! W' F3 w& e
15.在定义数组时,将定义的“元素个数”误认为是可使的最大下标值。
( [5 G1 ^5 @) Qmain(): z" m9 D7 V+ ?  J1 ~
{static int a[10]={1,2,3,4,5,6,7,8,9,10};
. x2 b; i" @0 t3 L0 z! eprintf("%d",a[10]);
' F5 u5 U2 `: m& L* M( _2 M" q}# Q' _2 u8 b/ p
C语言规定:定义时用a[10],表示a数组有10个元素。其下标值由0开始,所以数组元素a[10]是不存在的。6 c2 P; E, ?2 R* I! ~
16.初始化数组时,未使用静态存储。
1 s0 j) L+ D0 P8 Z0 B$ D! rint a[3]={0,1,2};
3 Z* ^1 ^" J& k/ F7 u! A) \6 m这样初始化数组是不对的。C语言规定只有静态存储(static)数组和外部存储(exterm)数组才能初始化。应改为:1 U3 g/ W9 o4 G8 Q( D
static int a[3]={0,1,2};
7 Y% S4 b( M6 u4 z4 \17.在不应加地址运算符&amp;的位置加了地址运算符。" r+ {5 k  y& Z5 G$ h. O
scanf("%s",&amp;str);
! @, l5 c2 w/ A# {' cC语言编译系统对数组名的处理是:数组名代表该数组的起始地址,且scanf函数中的输入项是字符数组名,不必要再加地址符&amp;。应改为:
9 }- X. E+ p& z0 ascanf("%s",str);
7 b+ }+ q! i3 r5 @18.同时定义了形参和函数中的局部变量。
8 H  {0 @8 E4 i: u. u" q1 J; N( R. rint max(x,y)
+ r, p* o( s: W. e1 z" Wint x,y,z;$ B! T5 K  P( i, N( b* e" a
{z=x&gt;y?x:y;: d2 C3 e0 D/ N5 k1 B
return(z);. o0 n' R* W' a* |! U0 u
}
' R" }, w$ a7 p: d$ \形参应该在函数体外定义,而局部变量应该在函数体内定义。应改为:
# B2 e5 R. h; w$ Gint max(x,y)1 P( X4 g/ }- G  H- n
int x,y;
1 g- B. n; z9 C7 X{int z;: r! S# i4 Z+ n+ f+ {7 l
z=x&gt;y?x:y;
- L' X2 x9 }6 M1 ^return(z);
$ p' A: N) T/ Y7 L! e- J" y}</P>- X% ^' M) y* B% {  e
<>以上错误中可能有些不符合新版的C语言,比如数组的初始化,新版中就可以是不是静态变量。由于是转贴就未加修改,以保持文章的原貌,请各位自加区别。<b>版权属于原创作者!!!</b></P>
zan
转播转播0 分享淘帖0 分享分享0 收藏收藏1 支持支持0 反对反对0 微信微信
笑一笑 只牵动苦涩的嘴角 我的寂寞谁知道 我害怕 用真心面对这世界 只好越来越沉默
您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-6-3 12:19 , Processed in 0.301196 second(s), 51 queries .

回顶部