keilMDK编译器警告和错误详解(不定期更新)
⼯作后从单⽚机转成ARM,刚开始⽤ADS1.2编译器,⽤了⼀段时间,因为我接⼿的项⽬的⽼程序正是⽤ADS编译的,部门也⼤都在⽤.在学单⽚机的时候⽤的是keil c51编译器,ads和这个编译器在易⽤性上真是⽆法⽐较.后来渐渐知道keil已经被arm公司收购,现在keil MDK 成为了arm官⽅编译器,所以决定重新投奔keil,利⽤平时的时间,将原程序重新⽤mdk编译.mdk的优点就没必要说了,在这⾥把平时遇到的编译器给出的警告和错误信息给出详解,希望给初学者⼀点帮助,发现错误,需要补充的欢迎留⾔.
1.warning: #550-D: variable "d" was set but never used
描述:变量'd'定义但从未使⽤,或者是,虽然这个变量你使⽤了,但编译器认为变量d所在的语句没有意义,编译器把它优化了.
解决:仔细衡量所定义的变量d是否有⽤,若是认定变量d所在语句有意义,那么尝试⽤volatile关键字修饰变量d,若是真的没有⽤,那么删除掉以释放可能的内存.
2.warning: #1-D: last line of file ends without a newline
描述:⽂件最后⼀⾏不是新的⼀⾏.编译器要求程序⽂件的最后⼀⾏必须是空⾏,想了半天没想通为什么要这样.
解决:可以不理会.若是觉得出现警告不爽,那么在出现警告的⽂件的最后⼀⾏敲个回车,空出⼀⾏.
3. warning: #111-D: statement is unreachable
描述:声明不可能到达.多出现在这种场合:
int main(void)
{
...
while(1) //⽆限循环,这在不使⽤操作系统的程序中最常见
{
...
}
return 0; //这句声明在正常情况下不可能执⾏到,编译器发出警告
}
解决:不理会.
4. warning: C3017W: data may be used before being set
描述:变量'data'在使⽤前没有明确的赋值.如:
uint8 i,data; //定义变量i和data,⼆者都没有明确赋值
for ( i = 0; i < 8; i++) //变量'i'在语句中被赋值0
{
if ( IO1PIN & SO_CC2420 )
data |= 0x01; //变量'data'在使⽤前没有明确赋值,编译器发出警告
else
data &= ~0x01;
}
解决:应仔细衡量该变量的初始值是否为0,若是,可以不理会这个警告,因为MDK编译器在程序执⾏前,会将使⽤到的数据区初始
化为0,但若是该变量的初始值不应该是0,忽略这个警告可能会引起致命错误.这个警告应引起⾜够重视.应养成变量赋初值的习惯,好在有编译器给把关.
5. warning: #177-D: variable "temp" was declared but never referenced
描述:变量'temp'进⾏了声明但没有引⽤.多出现在声明了⼀个变量,但却没有使⽤它,它和warning: #550-D: variable "temp" was set but never used不同之处在于temp从没有使⽤过.
解决:若是定义的变量确实没有⽤,删除掉;若是有⽤,则在程序中使⽤.
与该警告类似的还有warning: #177-D: function "MACProcessBeacon" was declared but never referenced
6. warning: #940-D: missing return statement at end of non-void function "DealwithInspect2"
描述:返回⾮空的函数"DealwithInspect2"的最后缺少返回值声明.如:
int DealwithInspect2(uint32 test)
{
...
...
...
//此处应该是return x;返回⼀个int型数据,若是没有返回值,编译器产⽣警告
}
7..warning: #1295-D: Deprecated declaration lcd_init - give arg types
描述:在定义函数的时候,如果你写上函数参数,就会有这个警告,⽐如void timer_init(); 这⾥就没有形参,如果这样的话,编译器会给出警告.
1. error: #65: expected a ";"
描述:缺少分号.⼤多是漏忘';'.
解决:双击错误⾏,在定位到错误点的附近到没加';'号的语句,加上分号.并不⼀定在定位到的错误⾏才却分号,可能是这⾏的上⼀⾏,也可能是下⼀⾏.
2. error: #65: expected a ";"和error: #20: identifier "xxxx" is undefined⼀块出现,⽽且后⾯的error: #20错误可能⼀⼤堆
描述:这个错误对于第⼀次遇上的⼈来说绝对是个噩梦,当错误出现,满怀希望的双击错误提⽰,来到错误⾏时却愕然发现,错误⾏绝对没有错,于是错误⾏的上⼀⾏,下⼀⾏,没有错误,再上上⾏,下下⾏...让⼈⽆⽐郁闷的事情出现了:编译提⽰的所有错误⾏都不可能有错误出现.其实这最可能是你在.h⽂件声明外部变量或者函数时,没有在声明语句的最后加分号!如果你有很多模块,如main.c,lcd.有很多头⽂件,如lcd.h,key.h,若是在lcd.h⽂件声明函数时没有加分号,那么这种错误可能定为到main.c中,所以要检查所有头⽂件.
解决:仔细检查.h⽂件,将分号补上.
3. Error: L6200E: Symbol flagu multiply defined (by uart0.o and main.o).
描述:变量(也是⼀种符号)flagu多处定义(在uart0.c中和main.c都定义了).通常错在全局变量定义重复.⽐
如:在main.c中定义全局变量flagu:
uint8 flagu=0;
在uart0.c中也⽤到该变量,于是声明此变量,我通常都是先复制定义的变量再在变量前⾯加关键字extern修饰:
extern uint8 flagu=0;
然后编译,就会出现上⾯的连接错误,原因在于,我在uart0.c中是⼜定义了⼀个变量,⽽不是声明变量,因为我给变量赋了初值"flagu=0",这样就重复定义了变量flag.正确的声明⽅法是去掉赋值部分:
extern uint8 flagu;
解决办法:到重复定义的变量,看情况修改⼀处.
<: #159: declaration is incompatible with previous "wr_lcd" (declared at line 40)
描述:在wr_lcd函数还没有声明之前就已经使⽤了.多出现在两种情况:第⼀种,wr_lcd函数体还没有写,就已经⽤到了它,这种情况多出现在写⼀个程序的⼤体结构中,只是简单写⼀下框架.第⼆种情况⽐较常见,函数a调⽤函数b,但函数b的函数体在函数a的下⾯:
void a(void) //函数a的实体
{
b(); //调⽤函数b
}
void b(void) //函数b的实体
{
...
}
这样如果点编译,就会产⽣error: #159的错误,因为当函数a调⽤函数b时,发现在这之前都没有函数b的任何声明.
编译器错误解决⽅法:在函数a调⽤函数b之前,对函数b进⾏声明,如:
void b(void); //对函数b进⾏声明
void a(void) //函数a的实体
{
b(); //调⽤函数b
}
void b(void) //函数b的实体
{
...
}
5. error: #137: expression must be a modifiable lvalue
描述:表达式必须是⼀个可以修改的左值.主要出现在这种现象:
a=NUM;
NUM是⼀个数值或表达式,a为⼀个变量,但a被定义为像const这种不可更改的类型,导致NUM不能赋值给变量a.
解决⽅法:要么放弃赋值,要么修改变量属性.
<: #18: expected a ")"
如果是出现在c⽂件中, 多半是因为少了⼀个")",或者错误⾏有编译器不识别的字符
如果出现在头⽂件中,错误⾏⼜是⼀个函数声明,多半是因为在函数声明中有编译器不认识的字符.
<: #7: unrecognized token
未识别的标记,多半是切换成了中⽂标点。