一年后再次回头看container_of宏的作用

本文作者回顾了一年前对container_of宏的分析,并指出当时的表述仍有改进空间。container_of宏用于从结构体成员的指针反推出整个结构体的地址。其工作原理包括使用typeof获取成员类型,将成员指针转换为结构体指针,并通过offsetof计算结构体首地址。文章简单介绍了offsetof宏的功能,并提供了container_of宏的使用方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

container_of宏在很久以前的文章我就已经有分析写过了,但是看回去,感觉以前自己对这个宏的表达描述分析能力还是有待提高,毕竟那时候刚接触linux内核不久,整个人都懵懵哒,无法说得让别人通俗易懂。

那么这个宏整体是什么样子的呢?

//获取结构体成员相对于结构体的偏移 
#define offsetof(TYPE,MEMBER) ((int) &((TYPE *)0)->MEMBER)  
//通过获取结构体中的某个成员,反推该结构体的指针 
#define container_of(ptr, type , member) ({ \
	const typeof(((type *)0)->member) *__mptr = (ptr) ; \
	(type *)((char *)__mptr - offsetof(type,member)) ;})
作用:通过某个结构体中的某个成员变量的指针,反推这个结构体变量的指针。说白了,我定义一个结构体变量和结构体指针,变量.结构体中某一个成员(这个我是知道的),还有知道它的地址,也就是&(变量.结构体中某一个成员),通过这两个参数,还有结构体的type,返回得到的就是结构体的首地址。

### 回答1: `container_of()` 是一个用于在一个数据结构中获取指向包含该结构的对象的指针的。它通常在内核开发中使用,特别是在实现链表或其他数据结构时使用。 在实现链表等数据结构时,通常会将数据结构嵌入到包含它的结构中。这种情况下,要获取指向包含它的结构的指针,可以使用 `container_of()` 。 例如,假设我们有一个包含一个 `struct list_head` 的结构体 `my_struct`: ``` struct my_struct { int data; struct list_head list; }; ``` 并且我们有一个指向 `list` 的指针 `struct list_head *ptr`,我们可以使用 `container_of()` 来获取指向 `my_struct` 的指针: ``` struct my_struct *p = container_of(ptr, struct my_struct, list); ``` 这个的实现非常巧妙,在实际使用时需要注意一些细节,比如数据结构中包含的成员变量名必须是 `list`,否则会编译错误。 ### 回答2: container_of是Linux内核中提供的一个非常重要的,并且在其他操作系统的内核开发中也有类似的实现。它的作用是通过指向结构体成员的指针来获取整个结构体的起始地址。 使用container_of可以方便地在提供结构体成员指针的情况下,快速地获取整个结构体的地址,从而可以方便地在后续的代码中访问结构体的其他成员。 container_of的使用场景非常广泛,特别在内核开发中经常使用。下面举几个常见的例子: 1. 链表操作中,可以通过链表节点的指针获取对应的结构体地址,进而进行各种链表操作。比如,在遍历链表时,可以通过container_of获取当前节点所在的结构体地址,并进行特定的操作。 2. 内核或驱动模块中,通过传递成员指针来获取整个包含该成员的结构体地址。在内核中,经常使用结构体指针作为参数传递,container_of可以方便地获取结构体的起始地址,并对整个结构体进行操作。 3. 内存分配中,container_of可以用于获取包含某个内存块的结构体地址。比如,在使用kmalloc分配内存时,可以通过container_of获取对应结构体的地址,并对其进行初始化。 总之,container_of作用及使用场景是为了在操作系统内核或驱动开发中,通过成员指针快速获取对应结构体的起始地址,从而方便地进行后续的操作。它的使用可以简化代码逻辑,提高开发效率,因此被广泛使用。 ### 回答3: container_of是Linux内核中的一个常用,它的作用是通过一个结构体成员变量的指针,推导出该结构体变量的起始地址。 在C语言中,结构体变量的不同成员在内存中是依次存放的。当我们拥有一个结构体成员变量的指针时,如果想获取这个指针所属的结构体变量的起始地址,就可以使用container_ofcontainer_of的定义如下: #define container_of(ptr, type, member) ({ \ const typeof(((type *)0)->member) *__mptr = (ptr); \ (type *)((char *)__mptr - offsetof(type, member)); }) 其中,ptr表示待推导的结构体成员变量的指针,type表示目标结构体类型,member表示成员变量名。 使用container_of时,我们需要提供三个参数:成员变量的指针,结构体类型和成员变量名。内部使用typeof关键字来获取成员变量的指针的类型,然后使用offsetof来获取成员变量在结构体中的偏移量。最后,通过将成员变量的指针减去偏移量,就可以得到结构体变量的起始地址。 container_of在Linux内核中广泛使用,在遍历链表的时候特别有用。由于链表的节点通常嵌入在结构体中作为成员变量,当我们拥有一个链表节点的指针时,使用container_of可以方便地获取到包含该节点的结构体变量的起始地址。 总的来说,container_of是Linux内核中的一个实用工具,通过结构体成员变量的指针,可以倒推结构体变量的起始地址。这在操作链表等场景中特别有用。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Engineer-Bruce_Yang

谢谢您

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值