新闻  |   论坛  |   博客  |   在线研讨会
Linux内核中container_of 和list_for_each_entry有什么区别?
嵌入式Linux | 2023-01-10 17:18:08    阅读:154   发布文章

container_of

功能:根据结构体中成员的地址从而获取到这个成员所在的结构体的首地址。


首先可以看出container_of被预定义成一个函数,三个参数,ptr:已知的结构体的成员的地址;type:结构体的类型;member:结构体成员的名字;函数的第一句代码,通过((type *)0)->member定义一个MEMBER型的指针__mptr,这个指针指向ptr参数,所以第一段代码获取到了结构体的成员member的地址;接下来我们用这个地址减去结构体成员member在结构体中的相对偏移量,就可以获取到所求结构体的首地址:(char *)__mptr - offsetof(type,member)就实现了这个过程,最后再把这个地址强制转换成type型指针,就获取到了所求结构体指针,define预定义返回最后一句话的值,将所求结构体指针返回。所以整个container_of的功能就是通过指向结构体成员member的指针ptr获取指向整个结构体的指针。container_of清楚了,那list_entry就更是一目了然了。


list_entry的功能等同于container_of。接下来分析list_for_each_entry的实现过程。

list_for_each_entry

Linux内核代码中,要经常对链表进行操作,list_for_each_entry是操作链表非常重要的一个宏函数:遍历给定类型的链表。


在理解了list_entry的基础上分析list_for_each_entry是一件比较轻松的事情,但在这里还是要强调一下双向链表及链表头的概念,否则对list_for_each_entry的理解还是一知半解。建立一个双向链表通常有一个独立的用于管理链表的链表头,链表头一般是不含有实体数据的,必须用INIT_LIST_HEAD()进行初始化,表头建立以后,就可以将带有数据结构的实体链表成员加入到链表中。list_for_each_entry被预定义成一个for循环语句,for循环的第一句话获取(head)->next指向的member成员的数据结构指针,也就是将pos初始化为除链表头之外的第一个实体链表成员,for的第三句话通过pos->member.next指针遍历整个实体链表,当pos->member.next再次指向我们的链表头的时候跳出for循环。整个过程没有对链表头进行遍历(不需要被遍历),所以使用list_for_each_entry遍历链表必须从链表头开始。因此可以看出,list_for_each_entry的功能就是遍历以head为链表头的实体链表,对实体链表中的数据结构进行处理。


*博客内容为网友个人发布,仅代表博主个人观点,如有侵权请联系工作人员删除。

参与讨论
登录后参与讨论
专注于嵌入式Linux知识的分享、交流
推荐文章
最近访客