接下来我们来了解一下计算结构体中成员变量关于结构体首地址的偏移量宏offsetof
#define offsetof(s, m) (size_t)&(((s *)0)->m)
理解方式一、
s是一个结构名,它有一个名为m的成员(s和m 是宏offsetof的形参,它实际是返回结构s的成员m的偏移地址.
(s *)0 是骗编译器说有一个指向类(或结构)s的指针,其地址值0
&((s *)0)->m 是要取得类s中成员变量m的地址. 因基址为0,这时m的地址当然就是m在s中的偏移
最后转换size_t 型,即unsigned int。
理解方式二、
首先将0强制类型转化为结构体指针类型s*,此时的零的类型就是 s*,那么其当然可以访问其成员m(通过(s*)0 -> m 的方式访问),那么此时再取这个变量的地址,即 & ((s*)0->m),这个地址呢减去结构体的基地址就是我们要求的偏移量。
下面我们来看个栗子。
#include<stdio.h>
#include<stdlib.h>
//#define offsetof(s, i) (size_t)&( ((s*)0)->i )
int main()
{
struct S1
{
char c1;
int i;
char c2;
}s;
printf("%d", offsetof(struct S1,c2));
system("pause");
return 0;
}