见代码原型:
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
#define container_of(ptr, type, member) ({
const typeof(((type *)0)->member) * __mptr = (ptr);
(type *)((char *)__mptr - offsetof(type, member)); })
例子:
typedef Struct Student
{
int id;
int age;
char name[32];
}Student;
分析:
offsetof
第一个参数:结构体的类型
第二个参数:结构体的成员的名字
原理:先把0地址转为传入的结构体类型,然后->MEMBER找到成员,最后取它的地址转为size_t类型就是结构体成员相对于结构体的偏移了, 因为结构体的起始地址是0,所以成员的地址就是相对于0地址的偏移。
使用:
int offset = offsetof(age, Student);
offset = 4;
container_of
第一个参数:结构体成员的指针
第二个参数:结构体的类型
第三个参数:结构体的成员
原理:先获取一个新的结构体成员的指针__mptr ,然后使用这个指针减去(结构体成员相对于结构体的偏移)offsetof宏,就可以得到结构体的指针了。然后再强制转换为type *就可以了。
使用:
Student S =
{
1,
20,
“Mark 666”,
};
Student * ps = container_of(&(S.age), Student , age)