|
|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
工作后从单片机转成ARM,刚开始用ADS1.2编译器,用了一段时间,因为我接手的项目的老程序正是用ADS编译的,部门也大都在用.在学单片机的时候用的是keil c51编译器,ads和这个编译器在易用性上真是无法比较.后来渐渐知道keil已经被arm公司收购,现在keil MDK成为了arm官方编译器,所以决定重新投奔keil,利用平时的时间,将原程序重新用mdk编译.mdk的优点就没必要说了,在这里把平时遇到的编译器给出的警告和错误信息给出详解,希望给初学者一点帮助,发现错误,需要补充的欢迎留言.
# q1 v( `. m9 }
, d5 o' u7 i# i' O9 V, P/ y1.warning: #550-D: variable "d" was set but never used% W% u+ R* x6 ] A
描述:变量'd'定义但从未使用,或者是,虽然这个变量你使用了,但编译器认为变量d所在的语句没有意义,编译器把它优化了.$ A9 I) d3 y; f5 @( s
解决:仔细衡量所定义的变量d是否有用,若是认定变量d所在语句有意义,那么尝试用volatile关键字修饰变量d,若是真的没有用,那么删除掉以释放可能的内存.2 v* h) [2 E8 ^
% k+ q* B# T2 h# `
2.warning: #1-D: last line of file ends without a newline
, d1 i4 c* r9 G. J! y6 T描述:文件最后一行不是新的一行.编译器要求程序文件的最后一行必须是空行,想了半天没想通为什么要这样.
1 l4 E; H& h. f解决:可以不理会.若是觉得出现警告不爽,那么在出现警告的文件的最后一行敲个回车,空出一行.7 U; w. G' r4 p1 }" F
$ s0 J6 @& b7 ^8 t; ?& J
3. warning: #111-D: statement is unreachable/ y: @5 m z' |) {4 F" {/ j
描述:声明不可能到达.多出现在这种场合:
. {8 [4 l) B* }4 sint main(void): b8 b6 ~5 f9 e3 K# A
{4 ^, J9 i7 k _
..., I5 Y7 C; o8 w1 _- u8 D3 v
9 F, _7 t0 D; U
while(1) //无限循环,这在不使用操作系统的程序中最常见
9 z4 |& @8 e" f7 |+ U( `8 q* M{( z: I9 ^ ~6 D* G/ T: \
...
9 A0 r* ]! y+ U" ]* C l" d8 W4 s2 C$ ^4 g+ c! p6 T- C1 ~
}
: _: I. o& k: L% _( l4 A6 X
/ F9 D7 C( H1 e. e+ ~0 hreturn 0; //这句声明在正常情况下不可能执行到,编译器发出警告- q+ e- E3 ? {7 s. `! q
}
3 u* S5 B, ?5 t# }# {
7 n; J9 w" k! l2 i9 Q解决:不理会.+ W7 C* [9 C! g3 Z
8 Y" e$ P/ R$ m5 g% i! B l! W/ c6 P4. warning: C3017W: data may be used before being set
& q! w7 A5 [) ^2 L描述:变量'data'在使用前没有明确的赋值.如:& N, b$ R- _" E) Z; y" z
uint8 i,data; //定义变量i和data,二者都没有明确赋值
' W6 V2 p" m6 s1 X
. G& C. @" c8 j' C3 V& I+ f, Hfor ( i = 0; i < 8; i++) //变量'i'在语句中被赋值02 _1 @) L8 G* D
{
( W2 O5 f* {/ ?) {! Kif ( IO1PIN & SO_CC2420 )% z. O5 Y5 t+ X+ L" `& ~+ n" V
data |= 0x01; //变量'data'在使用前没有明确赋值,编译器发出警告6 \/ V) \- r# e5 i
else; B& ~- b/ ]- T* N8 e
data &= ~0x01;* q3 }' X& Q: M1 u1 Q4 Q1 P
}
' r' F& W4 |1 d8 d1 p. G2 F. _, v解决:应仔细衡量该变量的初始值是否为0,若是,可以不理会这个警告,因为MDK编译器在程序执行前,会将使用到的数据区初始化为0,但若是该变量的初始值不应该是0,忽略这个警告可能会引起致命错误.这个警告应引起足够重视.应养成变量赋初值的习惯,好在有编译器给把关.
. g# ^, _, ^# X% \) w( j+ q. N# M* W6 n
5. warning: #177-D: variable "temp" was declared but never referenced
" ^. I i8 f0 {- [4 m' }+ ~+ g描述:变量'temp'进行了声明但没有引用.多出现在声明了一个变量,但却没有使用它,它和warning: #550-D: variable "temp" was set but never used不同之处在于temp从没有使用过.: L9 H5 m& [7 N u, }( i
解决:若是定义的变量确实没有用,删除掉;若是有用,则在程序中使用.
; E% H3 x# \" M! p与该警告类似的还有 warning: #177-D: function "MACProcessBeacon" was declared but never referenced+ L- J# D' ~- o8 w" q/ n8 l
- r( o& Z- v- V3 ?: ?: b4 R( `
6. warning: #940-D: missing return statement at end of non-void function "DealwithInspect2"% q3 q7 X0 |+ m9 A: r) @9 |% Y4 d
描述:返回非空的函数"DealwithInspect2"的最后缺少返回值声明.如: L5 B) [/ n" [) a# q
int DealwithInspect2(uint32 test), C& x G; u) h
{; A1 i, M+ i. N
...! c$ x6 L0 F( r8 \6 a% p
...7 J/ N9 w1 z! p1 v9 ^/ u
..., ?$ w0 K3 Q* a6 U- n R% y
//此处应该是return x;返回一个int型数据,若是没有返回值,编译器产生警告; i& D/ g( y; u' W3 j
}% ]+ J: f1 D' c6 T3 P- F4 l
) H( n9 F: `$ _9 I4 `. \# ]* {
7.
4 X2 D) ^# s4 ]- Z' h
1 T4 L4 [; S$ k& y/ H6 y' ]1. error: #65: expected a ";"0 d* h- i) _6 F T$ R( i
描述:缺少分号.大多是漏忘';'.
3 o+ p; V# i: i+ h. h, K: ]! Z0 v+ _解决:双击错误行,在定位到错误点的附近找到没加';'号的语句,加上分号.并不一定在定位到的错误行才却分号,可能是这行的上一行,也可能是下一行.0 _) n; N9 t' k& i7 j
% X0 V' L+ N) f4 ^; w
2. error: #65: expected a ";"和 error: #20: identifier "xxxx" is undefined一块出现,而且后面的error: #20错误可能一大堆
) I! N1 B) o, j" j描述:这个错误对于第一次遇上的人来说绝对是个噩梦,当错误出现,满怀希望的双击错误提示,来到错误行时却愕然发现,错误行绝对没有错,于是找找错误行的上一行,下一行,没有错误,再找上上行,下下行...让人无比郁闷的事情出现了:编译提示的所有错误行都不可能有错误出现.其实这最可能是你在.h文件声明外部变量或者函数时,没有在声明语句的最后加分号!如果你有很多模块,如main.c,lcd.c,key.c...有很多头文件,如lcd.h,key.h,若是在lcd.h文件声明函数时没有加分号,那么这种错误可能定为到main.c中,所以要检查所有头文件.
$ u- ?( Y: ~- R: W解决:仔细检查.h文件,将分号补上.8 z G+ \8 e. v$ R$ w* E: t
3 ~( |. z8 H% f W8 Q% V. a
3. Error: L6200E: Symbol flagu multiply defined (by uart0.o and main.o).! q' P& T0 o1 u% ]* D* a+ y
: [' m6 U4 [$ E8 T' s8 n+ n: P p! P+ g
描述:变量(也是一种符号)flagu多处定义(在uart0.c中和main.c都定义了).通常错在全局变量定义重复.比如:在main.c中定义全局变量flagu:
1 p6 I& {9 v# g0 G' f/ j8 W4 p7 y) x0 t8 }9 y. D
uint8 flagu=0;
' s- W6 t* k+ O6 ?% s/ ]1 I6 k, J% y" d, G
在uart0.c中也用到该变量,于是声明此变量,我通常都是先复制定义的变量再在变量前面加关键字extern修饰:
+ P$ P% D1 Q9 h1 H) o$ e8 X
) l- I2 V$ [ P1 E+ y" t9 U3 |extern uint8 flagu=0;- }9 V) I; n5 d0 k
5 q8 H2 `( Y6 U+ F. O
然后编译,就会出现上面的连接错误,原因在于,我在uart0.c中是又定义了一个变量,而不是声明变量,因为我给变量赋了初值"flagu=0",这样就重复定义了变量flag.正确的声明方法是去掉赋值部分:' B' T9 R" S, a' Z0 E% T' x
4 Q. Y- l, F- qextern uint8 flagu;) L v/ y$ @" ^+ g2 ]
- S. x( w' H- ^+ }+ R, k- F
解决办法:找到重复定义的变量,看情况修改一处.
0 }& W9 Z7 p& K# }: ?2 r2 G* A: b* b4 J: J7 r! g9 c
4.error: #159: declaration is incompatible with previous "wr_lcd" (declared at line 40)
3 M( L" i z2 H& }描述:在wr_lcd函数还没有声明之前就已经使用了.多出现在两种情况:第一种,wr_lcd函数体还没有写,就已经用到了它,这种情况多出现在写一个程序的大体结构中,只是简单写一下框架.第二种情况比较常见,函数a调用函数b,但函数b的函数体在函数a的下面:% E& v3 _% ^" `( A/ w$ N/ N. S
void a(void) //函数a的实体
. S0 R2 @2 _ I{' j' t9 A h# t* J9 z6 K4 v
b(); //调用函数b) F. S& A6 L) [" Q; W" r
}: _4 k! { l: M! N. {* I
- Z% `+ P1 I( [: \void b(void) //函数b的实体. C5 {, B% f5 `. `# T3 T
{# c" F& I5 O/ K
...2 ~$ F x7 N- B1 P/ G
}
" h( g, {, |8 L8 Q4 j2 D( [# |! A4 g这样如果点编译,就会产生error: #159的错误,因为当函数a调用函数b时,发现在这之前都没有函数b的任何声明.1 v, B$ I% G I6 S
解决方法:在函数a调用函数b之前,对函数b进行声明,如:! }3 E0 ]( f5 z( a0 R
void b(void); //对函数b进行声明) V2 t( G1 |% k" D" C* P0 d3 [% ~
9 z) P O P, `void a(void) //函数a的实体
: m; w6 E$ ?! V0 l5 D{; A' [2 c; I$ u6 o" ?! ]
b(); //调用函数b5 [; \4 O, h6 _$ A n: t2 B6 o
}
/ P$ d, Y- D( V% ^1 m5 a; N, ?4 T3 t0 [
void b(void) //函数b的实体
- t( I4 M( k3 S6 R* s* Y3 a5 _3 c{
* O* H* [) }8 }3 w( x) X...! S1 `) }0 C1 {& d! l. W
}7 |" U1 M3 k" u- ~" L- A+ f
|
|