目录
- 一.offsetof宏定义的使用
- 二.offsetof宏的模拟实现
- 三.总结
一.offsetof宏定义的使用
offsetof的第一个参数是数据类型(结构体或联合体),第二个参数是偏移量
(offsetof用于计算结构体和联合体的偏移量)
返回值的类型是size_t(无符号整形)
offsetof要包含的头文件stddef.h
因为偏移量是某个地址距离0地址处的大小,
所以肯定是一个正数
下面举一个例子:
struct A
{
char a;
int b;
};
int main()
{
printf("%zd", offsetof(struct A, a));//0
//%zd表示size_t类型,
printf("%zd", offsetof(struct A, b));//4
//对齐规则不知道的可以看我的结构体那篇
//计算的都是这个元素的第一个字节的地址距离0地址处的大小
return 0;
}
二.offsetof宏的模拟实现
#define MY_offsetof(s,m) ((size_t)&(((s*)0)->m))
思路:
s是结构体类型,将0地址强转为结构体指针
0然后指向结构体成员名,就找到了结构体成员
整体&,就是结构体成员变量的地址(指针)
假如m是int,然后&,int 强转为size_t类型要看平台大小吗?*
其实这个问题是错的,->拿到的是成员变量的地址
是对于0地址处的大小
强制类型转化为size_t类型就是偏移量大小
#define MY_offsetof(s,m) ((size_t)&(((s*)0)->m))
struct A
{
char a;
int b;
};
int main()
{
printf("%zd\n", MY_offsetof(struct A , a));//0
printf("%zd\n", MY_offsetof(struct A , b));//4
return 0;
}
三.总结
这个宏的模拟实现主要还是不好想到
0地址处强转为结构体指针指向成员变量m
然后&拿到成员变量的地址,对于0地址处的偏移量