从一个成员得到结构体的地址

本文详细解释了container_of宏的工作原理及其在Linux内核中的应用。通过对宏定义的深入剖析,揭示了如何通过成员变量地址来定位到所属结构体变量的过程。
container_of宏

#define offset_of(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
TYPE  // 结构名
MEMBER // TYPE 结构中的成员
(TYPE *)0 // 告诉编译器有一个指向 TYPE 结构的指针,其值为止0
((TYPE *)0)->MEMBER // 取结构 TYPE 中的 MEMBER 成员
&((TYPE *)0)->MEMBER // 取结构 TYPE 中的 MEMBER 成员的地址

因为基地址为0,所以这时 MEMBER 成员的地址就是 MEMBER 在 TYPE 中的偏移
最后再把结果强制转换为size_t型   
作用: 返回结构 TYPE 中的成员 MEMBER 的偏移地址
offsetof宏定义在[include/linux/stddef.h]中

#define container_of(ptr, type, member) ({               /
    const typeof( ((type *)0)->member ) *__mptr = (ptr); /
    (type *)( (char *)__mptr - offset_of(type,member) );})



container_of宏定义在[include/linux/kernel.h]中

typeof() 是 gcc 的扩展,和 sizeof() 类似 ,以获得 member 成员的数据类型
这里使用的是一个利用编译器技术的小技巧,即先求得结构成员在与结构中的偏移量,然后根据成员变量的地址反过来得出属主结构变量的地址。
typeof( ((type *)0)->member ) // 获得 member 成员的数据类型
const typeof( ((type *)0)->member ) *__mptr // 声明一个const 指针

在C语言中,有多种方式可以获取结构体成员地址,同时涉及一些相关知识。 ### 获取结构体成员地址的方法 - **获取结构体成员偏移地址**:可以通过`#define ptr &(((stud*) 0)->score)`来获取结构体成员的偏移地址,该偏移地址以字节为单位,并非成员结构体中的顺序。这里`((stud*) 0)`将0地址处的一段内存表示成`stud*`型的内存,`&(((stud*) 0)->score)`得到该成员的绝对地址,由于结构体地址为0,所以此值就是该成员在本结构体内存中的偏移值。不过需要将指针转化为`void*`类型进行运算 [^2]。 - **获取结构体的绝对地址**:通过实际寻找结构体成员绝对地址减去偏移地址可以得到本结构体的绝对地址,如`stud * stru_ptr = mem_ptr - (void *)ptr` ,但要注意最后进行相应的数据类型转化 [^1]。 ### 相关知识 - **结构体内存特性**:所有的结构体在C语言程序中只要初始化后都占有一定大小的内存,对于同一结构体而言,其包含的各结构体成员内存大小固定,即相对于结构体地址的偏移量固定 [^1]。 - **应用场景**:在嵌入式工作中,已知要获取信息的flash地址时,为了方便快捷地从flash中取出信息元素,时常需要知道结构体相对于已知地址的偏移 [^3]。 ### 示例代码 ```c #include <stdio.h> // 定义一个示例结构体 typedef struct { int id; float score; } stud; #define OFFSET_SCORE &(((stud*) 0)->score) int main() { stud s; // 获取成员score的绝对地址 float *mem_ptr = &s.score; // 获取结构体的绝对地址 stud *stru_ptr = (stud *)((void *)mem_ptr - (void *)OFFSET_SCORE); printf("结构体的绝对地址: %p\n", (void *)stru_ptr); printf("成员score的绝对地址: %p\n", (void *)mem_ptr); return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值