关于函数宏offset_of 和 container_of的学习
#define offset_of(type, member) ((unsigned int)&((type*)0)->member) #define container_of(ptr, type, member) ((type *)((char *)(ptr) - offset_of(type, member)))
offset_of(type, member)
用途: 用于获取获取结构体某一个成员在该结构体中的位置
参数1:type ,表示结构体的类型
参数2:member 表示结构体成员
分析:
(unsigned int) & (type*)0)->member a.把值为0的指针强制转换成该结构体类型
b.通过该指针找到该成员
c.获取该成员相对于0 的地址偏移
d.强转成整形
container_of(ptr, type, member)
用途: 知道结构体中某一个成员的地址,需要获取到整个结构体的指针,通过该指针获取其他成员变量的数据
参数1:ptr,某一个成员的地址
参数2:type 结构体类型
参数3:member 该成员的名字
分析:
(type *)((char *)(ptr) - offset_of(type, member))
a.获取该成员相对于结构体指针的偏移
b.将该成员的地址转换成char*类型,该指针指向1byte的数据
c.转换成结构体类型的指针
d.使用该结构体成员的指针 - 该成员相对于结构体指针的偏移 = 结构体指针
注意事项:
使用gcc编译的时候有遇到此问题:warning: cast from pointer to integer of different size [-Wpointer-to-int-cast]
这是一个编译器警告,指示在某一行代码中进行了从指针到整数的类型转换,而且这两者的大小不同。
(unsigned int) & (type*)0)->member 中,由于我是在32位单片机中使用,所以unsigned int与地址长度相同。
如果在其他不同的系统中需注意该指针的强制转换问题,可以使用
uintptr_t来解决跨平台的问题。
该类型定义在c库<stdint.h>中