今天同事问了我一个问题,他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