关于编译报错“dereferencing pointer to incomplete type...

今天同事问了我一个问题,他make的时候报错,“第201行:dereferencing pointer to incomplete type”,我随即查阅了不少资料,也没看出个因此然。最后问题获得了解决,也懂得了原理,遂记录一下。测试

他的问题具体是这样。spa

#include <netinet/ip_icmp.h>
...
struct icmp* aaa;
    aaa = (struct icmp*)malloc(sizeof(struct icmp)); //假设是第200行;
    aaa->icmp_type=1; //假设是201行;
...

make的时候第201行报错:dereferencing pointer to incomplete type。code

首先说一下这个报错的意思,通俗的说就是,试图访问该pointer指向的变量,却发现该变量是一个不完整的类型,多出错于访问结构体联合体的成员。从代码中可看出来,是从201行开始才真正的访问icmp_type指向的变量,200行还没访问。递归

因而我就猜测,是否是struct icmp没有定义呢?遂粗略的查看了/usr/include/netinet/ip_icmp.h文件,发现有struct icmp的定义。很奇怪,不是吗?通过写了一些demo测试,最终的结论是,确实没有struct icmp的定义!ip

看到这里,更奇怪了。为何是这样的结论呢?细看/usr/include/netinet/ip_icmp.h文件,会发现struct icmp的定义被包含在一个宏里面j,以下面所示:ci

...
#ifdef __USE_BSD
...
struct icmp {
...
}
...
#endif /*END OF ifdef __USE_BSD*/
...

看到这里应该就明白了,编译的时候,若是编译命令 gcc ...里面没有加入 -D__USE_BSD的话,那么struct icmp的定义是不会被include进来的,因此就致使了前面的第201行报错:dereferencing pointer to incomplete type,也就是这样致使我开始一直想不明白,明明有定义,为什么却说是不完整的类型。因而为了验证这个结论,我写了一个小demo来测试,发现加 -D__USE_BSD就编译经过,不然就编译不经过。编译

在解决这个问题的过程当中,我写了很多demo,,下面总结一下。class

1.若是报错“dereferencing pointer to incomplete type”,先试图找一下该行的那个结构体变量的定义是否能找到,可以使用grep "struct xxx" /usr/include -R命令递归搜索/usr/include目录,如找到,可在.c文件中#include,若是是非标准头文件就要在编译命令中加入-I头文件目录,例如(-I/usr/local/xxx/include)。变量

2.若是#include以后仍然报错“dereferencing pointer to incomplete type”,试图仔细查看该文件,查看该结构体的定义是否被某个编译宏给包裹了,若是确实处于某个编译宏的包裹内,在编译命令里面增长 -D编译宏(如-D__USE_BSD)原理

通过上面两个步骤之后,基本上能解决“dereferencing pointer to incomplete type”报错了。

2012年7月20日17:01:59

相关文章
相关标签/搜索