typeof 返回变量的类型,c语言32个关键字之一
offsetof 返回成员相对于结构体的偏移地址
#define (, ) (() &(( *)0)->)
1. ( (TYPE *)0 ) 将零转型为TYPE类型指针;
2. ((TYPE *)0)->MEMBER 访问结构中的数据成员;
3.
&( ( (TYPE *)0 )->MEMBER
)取出数据成员的地址;
4.(size_t)(&(((TYPE*)0)->MEMBER))结果转换类型;
巧妙之处在于将0转换成(TYPE*),结构以内存空间首地址0作为起始地址,则成员地址自然为偏移地址。
(注意括号的使用)
container_of
从结构体(type)某成员变量(member)指针(ptr)来求出该结构体(type)的首指针。
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
* @ptr: the pointer to the member.
* @type: the type of the container struct this is embedded in.
* @member: the name of the member within the struct.
1.typeof(
( (type *)0)->member
)为取出member成员的变量类型。用其定义__mptr指针.ptr为指向该成员变量的指针。__mptr为member数据类型的常量指针,其指向ptr所指向的变量处。
2.(char
*)__mptr转换为字节型指针。(char *)__mptr - offsetof(type,member))用来求出结构体起始地址(为char
*型指针),然后(type *)( (char *)__mptr -offsetof(type,member) )在(type
*)作用下进行将字节型的结构体起始指针转换为type *型的结构体起始指针,(
地址通过字节进行加减,指针-数字实际为指针 - 数字*sizeof(该指针类型))。
3.({ })这个扩展返回程序块中
最后一个表达式的值。
这就是从结构体某成员变量指针来求出该结构体的首指针。指针类型从结构体某成员变量类型转换为该结构体类型。
(type *)0->member为设计一个type类型的结构体,起始地址为0,编译器将结构体的起始的地址加上此结构体成员变量的偏移得到此结构体成员变量的偏移地址,由于结构体起始地址为0,所以此结构体成员变量的偏移地址就等于其成员变量在结构体内的距离结构体开始部分的偏移量。即:&(type *)0->member就是取出其成员变量的偏移地址。而其等于其在结构体内的偏移量:即为:(size_t)(& ((type *)0)->member)经过size_t的强制类型转换后,其数值为结构体内的偏移量。该偏移量这里由offsetof()求出。
阅读(819) | 评论(0) | 转发(0) |