合肥做双语网站,做网站的人属于什么行业,长沙58同城招聘网最新招聘,杭州做百度推广的公司一、前言昨天在编译代码的时候#xff0c;之前一直OK的一个地方#xff0c;却突然出现了好几个 Warning!本着强迫症要消灭一切警告的做法#xff0c;最终定位到#xff1a;是结构体内部#xff0c; 指向结构体类型的指针成员变量导致的问题。这个问题#xff0c;也许永远… 一、前言昨天在编译代码的时候之前一直OK的一个地方却突然出现了好几个 Warning!本着强迫症要消灭一切警告的做法最终定位到是结构体内部 指向结构体类型的指针成员变量导致的问题。这个问题也许永远不会碰到之所以被我赶上了应该是因为某个时候手贱 误碰了键盘导致。下面一一道来。PS: 我的测试环境是 Ubuntu16.04-64编译器使用系统自带的 gcc-5.4.0。二、问题描述1. 正常的代码比较简单结构体 struct _Data2_ 的第 2 个成员变量是一个指针指向的数据类型是结构体struct _Data1_。typedef struct _Data1_
{int a;
}Data1;typedef struct _Data2_
{int b;struct _Data1_ *next;
}Data2;int main()
{Data1 d1 {1};Data2 d2 {2, d1};printf(d1 %p \n, d1);printf(d2 %p \n, d2);}编译、执行都没有问题$ gcc main.c -m32 -o main
$ ./main
d1 0xffdc72f0
d2 0xffdc72f42. 错误的代码现在我们来模拟误碰键盘操作把 struct _Data2_ 中 next 成员指向的数据类型改为一个不存在的结构体typedef struct _Data2_
{int b;struct _Data3_ *next;
}Data2;在测试代码中struct _Data3_ 肯定是不存在的。好了现在执行编译指令 gcc main.c -m32 -o main将会得到什么结果可以停下来稍微 思考一下。我之前的预期是gcc 会 报错找不到 struct _Data3_ 这个类型。实际情况是$ gcc main.c -m32 -o main -I./
main.c: In function ‘main’:
main.c:18:20: warning: initialization from incompatible pointer type [-Wincompatible-pointer-types]Data2 d2 {2, d1};^
main.c:18:20: note: (near initialization for ‘d2.next’)
$ ./main
d1 0xffd8ee70
d2 0xffd8ee74好神奇吧 gcc 居然不报错那么我们就按照 gcc 的方式来理解一下。我们知道编译器在遇到一个结构体类型的时候最重要的就是需要知道结构体类型 所占据的内存空间的大小。gcc 在遇到 struct _Data2_ 这个字符串时判断出它是一个用户自定义的数据类型结构体 _Data2。gcc 继续读取结构体内部的每一个字符在读取到 *next 时知道它是一个 指针。此时它并并没确认该指针所指向的数据类型是否存在它只是为 next 保留了 4 个字节的内存空间(32位系统)。然后 gcc 在解析 Data2 d2 {2, d1}; 这一行时就发现 类型不匹配了data2 的 next 需要的是 struct _Data3_ 类型的指针但是赋值的 d1 是 struct _Data1_ 类型于是给出警告信息。我们用其他的编译器试一下(1) clang$ clang main.c -m32 -o main -I./
main.c:18:20: warning: incompatible pointer types initializing struct _Data3_ * with an expression of type Data1 *(aka struct _Data1_ *) [-Wincompatible-pointer-types]Data2 d2 {2, d1};^~~
1 warning generated.
$ ./main
d1 0xffb1b3a0
d2 0xffb1b398(2) g $ g main.c -m32 -o main -I./
main.c: In function ‘int main()’:
main.c:18:23: error: cannot convert ‘Data1* {aka _Data1_*}’ to ‘_Data3_*’ in initializationData2 d2 {2, d1};看起来只有 g 进一步确认了 _Data3_ 这个结构体类型不存在三、把类型改为 void 指针类型把 struct _Data2_ 中的 next 成员改为 指向 void 型的指针然后在 main 函数中操作它。typedef struct _Data1_
{int a;
}Data1;typedef struct _Data2_
{int b;void *next;
}Data2;int main()
{Data1 d1 {1}; Data2 d2 {2, d1};Data1 *dn d2.next;printf(dn-a %d \n, dn-a);
}编译、执行$ gcc main.c -m32 -o main -I./
$ ./main
dn-a 1可以看到Data1 *dn d2.next; 这一行把指向 void 型的 d2.next 赋值给指向Data1型的指针变量 dn然后在 printf 语句中可以正确地打印出dn中的成员变量a。这又回到了指针的本质 指针就是一个地址至于如何来解释这个地址中的内容这是由定义这个指针时所指定的数据类型来决定的结合代码来看虽然d2.next是一个 void 型指针但是它的确存储了一个 地址变量 d1 的地址。然后把这个地址赋值给dn 指针那么通过dn指针来操作该地址内的成员时就取决于在定义dn时所指定的数据类型(Data1)因此 dn-a 就可以正确的从这个地址中取出前 4 个字节然后作为一个int型的数据打印出来。以上代码如果使用clang来编译结果也是正确的。用g 编译继续报错$ g main.c -m32 -o main -I./
main.c: In function ‘int main()’:
main.c:23:20: error: invalid conversion from ‘void*’ to ‘Data1* {aka _Data1_*}’ [-fpermissive]Data1 *dn d2.next;如果想让这个错误消除掉在指针赋值时 强制转换一下即可(把void型指针强转成Data1型指针然后再赋值)Data1 *dn (Data1 *)d2.next;四、总结这里描述的错误几乎很少遇到除非是像我一样误碰了键盘。不过从中我们也看到了一个现象gcc编译器在面对结构体时主要关心的是结构体在内存空间中所占用的空间大小对其内部指向结构体类型的指针并没有严格的检查是否存在g 在这一点就做的严谨一些了。---------- End ----------声明本文于网络整理版权归原作者所有如来源信息有误或侵犯权益请联系我们删除或授权事宜。