概述:
我们之前说过使用->运算符获取结构体内部元素的偏移值(结构体内部第一个元素的偏移值为0),我们还可以使用标准库中提供的offsetof宏来获取结构体中数据的偏移。
需要导入头文件#include <stddef.h>
typedef struct type {
char c;
int a;
char b;
}Type;
printf("%d\n", (int)&(((Type*)0)->b)); // 8
printf("%zd\n", offsetof(Type, b)); // 8
上面使用两种方式获取了结构体中元素b在结构体中的偏移值。
offsetof() 第一个参数为 结构体类型, 第二个参数为要查看的偏移值的数据。 返回一个size_t的无符号整数,表示对应数据在结构体的偏移值。
那么为什么偏移值为8呢?
因为,结构体中数据的偏移是从0开始的,而且为了cpu更快的检索数据,会进行内存对齐,就是结构体中最大的类型其偏移值必须为其类型占有字节的整数倍,结构体最后占有的字节数应该是其元素最大类型的整数倍。
所以,此处Type中int类型为最大的类型,占用4字节,所以其偏移值必须是4的整数倍,所以这时候就会发生内存对齐,char本来占用1个字节,发生内存对齐之后,为4字节,偏移就是0-3,因为结构体数据是连续存储,所以接下来的a的偏移就是4,这样其的偏移值就是4的整数倍了,又因为此类型最后一个类型为char,值占用一个字节,这样占用的总字节数为4+4+1 = 9 ,并不是4的倍数,所以又会发生内存对齐,对齐到12,这时候就是4的倍数了,所以这个结构体占用的空间是12个字节。