16种C语言编译警告(Warning)类型的解决方法

所属分类: 软件编程 / C 语言 阅读数: 107
收藏 0 赞 0 分享

当编译程序发现程序中某个地方有疑问,可能有问题时就会给出一个警告信息。警告信息可能意味着程序中隐含的大错误,也可能确实没有问题。对于警告的正确处理方式应该是:尽可能地消除之。对于编译程序给出的每个警告都应该仔细分析,看看是否真的有问题。只有那些确实无问题的警告才能放下不管。

说明:

由于编译的警告各种各样,根本不可以一一罗列出来,下面只是列举出比较典型的一些警告,还有一些警告,大家只要根据字面意思,就可以很快的查找出来,并解决之。

类型1:
显示:warning: implicit declaration of function 'Example()'。

警告原因:

在你的.c文件中调用了函数Example(),可是你并没有把声明这个函数的相应的.h文件包含进来。
有可能你在一个.c文件中定义了这个函数体,但并没有在.h中进行声明。
解决方法:

你可以在调用这种函数的.c文件的一开始处加上:extern Example();
你可以在调用这种函数的.c文件中包含进声明了函数Example()的头文件。
如果你在一个.c文件中定义了这个函数体,但并没有在.h中进行声明,不嫌麻烦的话,你也可以去生成一个.h文件,加上你的函数声明。
类似的警告:

warning: type mismatch with previous implicit declaration
warning: type mismatch with previous implicit declaration
warning: previous implicit declaration of 'Example()'
 
类型2:
显示:warning: unused variable 'param'。

警告原因:很明显,是您定义了变量‘param',却根本没有使用它。

解决方法:不需要用的话,就删了它吧。

类型3:
显示:warning: statement with no effect。

警告原因:可能的情况是,在你的文件中,你这么干#define MACROPRINT

然后在某一处又定义了#define MACROPRINT printf。然后你在各处引用

MACROPRINT(“HELLO”),这样不会出错,但是发生了警告“这个声明是没有用的”。

解决方法:把#define MACROPRINT删掉。

类型4:
显示:warning: int format, long int arg (arg 3)

警告原因: 象这样printf("%s%d, szDebugString, ulGwId);你的ulGwId是一个unsigned long型的,而你为它选择的输出形式却是 “%d”(这个格式是为整数型服务的-int)。

解决方法: 这样的错误你只要做到参数类型一致就可以了,象上面的现象,你只要把“%d”改成“%d”就可以了。

类似警告:warning:comparison between pointer and integer

类型5:
显示:warning: comparison is always 0 due to limited range of data type

警告原因:有可能你定义了unsigned int uParam;但是你去做了if(uparam<0)的判断,

因 为unsigned int型的数据总是>=0的,因此这样的比较由于数据类型限制了它的范围,因此也就给出了警告。

解决方法:可以去掉这样的判断。

类型6:
显示:warning: control reaches end of non-void function

警告原因: 出现这样的警告,有可能是你写了一个

unsigned long FuncA()
{
  if()
  {
    return ulValue;
  }
  if()
  {
    return ulValue;
  }
}

这样的函数,可能在两个if语句中,你都没有进入,这时,退出函数之前,你就根本没有值可以返回。

解决办法: 如果一个函数有返回值,确保在任何情况下该函数都有一个返回值。

类似警告: warning :'return' with no value, in function returning non-void

类型7:
显示:warning: overflow in implicit constant conversion

警告原因:变量的变换有可能导致数值的越界。

#define RET_PRODUCTID 0x10000000
#define ERR_RET_GLOBAL RET_PRODUCTID+5000
#define RET_USER ERR_RET_GLOBAL+5000
#define USER_OK RET_USER+0
#define USER_FAIL RET_USER+1

如果这样定义,碰到short Func(){return USER_OK},就会警告有出现越界。

解决办法:确定好值的范围。

类型8:
显示:warning: 'ulParam' might be used uninitialized in this function

警告原因: 当ulParam做为表达式的右值时,而在此之前,你又没有对这个参数进行初始化。

例如:

void Func()
{
  ulong ulParam;
  ulong ulRetCode;
if(…)
{
  ulParam = ……;
}
if(….)
{
  ulParam = ……;
}
  ulRetCode = ulParam;
}

在这种情况下,当两个if()都执行不到的时候,ulParam根本没有被赋值过,这样又去给ulRetCode赋值,就比较危险了。

解决办法: 多留个神,细心一点就可以了。

类型9:
显示: warning: passing arg 1 of 'free' makes pointer from integer without a cast

警告原因: 你free(a),但a是一个unsigned long,你可能把一个指针的数值放在了a里面了。

解决办法: 在free(a)时,需要强制转换a为指针类型的即可。即:free((char*)a)。

类似警告:warning: assignment from incompatible pointer type

warning: initialization from incompatible pointer type

warning:passing arg 2 of 'AOS_MemCopy_X' makes pointer from integer without a cast

类型10:
显示: warning: 'MY_DEBUG' redefined

warning: this is the location of the previous definition

警告原因: 连续出现这种两个警告,可能的一种情况是,你在你的.c文件中包含了两个.h

文件,而这两个.h文件都对MY_DEBUG进行了声明。

解决办法:只在一个文件中声明这种东东。

类型11:
显示:warning: value computed is not used

警告原因:参与运算的值是没有作用的。比如你这样干:

char* p;

*p++;

这样对p根本一点影响也没有。

解决方法:请确定究竟要进行什么运算。

类型12:
显示:warning: '#ifdef' argument starts with a digit

警告原因:出现了#ifdef 0这样的错误

解决方法:应该是#if 0 吧

类型13:
显示:warning: unknown escape sequence '\R'

警告原因:编译器不认识‘\R'。

解决方法:一时笔误,应该是‘\r'。

类型14:
显示:warning:too few arguments for format

警告原因:你有可能这样干了:printf(“%d%s”,uParam);

解决方法:把要的留下,不要的去掉。

类型15:
显示:warning: ‘Func' defined but not used

警告原因:Func 函数你定义了,但是你根本没有使用它。

解决方法:不要的就去掉。

类型16:
显示:warning: suggest parentheses around && within ||

警告原因:有人这么用了

if(( *p >= 'a' ) && ( *p <= 'z' ) || ( *p >= 'A' ) && ( *p <= 'Z' ) || ( *p >= '0' ) && ( *p <= '9' ))

解决方法:你最好这样

if((( *p >= 'a' ) && ( *p <= 'z' )) || (( *p >= 'A' ) && ( *p <= 'Z' )) || (( *p >= '0' ) && ( *p <= '9' )))

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多