list.h分析1.docxVIP

  1. 1、本文档共4页,可阅读全部内容。
  2. 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  5. 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  6. 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  7. 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  8. 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
list.h分析1

list.h分析 暂存 2011-06-21 21:58 list.h所在位置: /usr/include/linux/list.h 其结构体为: struct list_head {   struct list_head *next, *prev;   }; struct list_head为一个空的双向连表。在很多用户自定义的结构体中都会将其添加进去,形成一个双向连表。 例如: struct my_list{ ?? ?int id; ?? ?char name[14]; ?? ?struct list_head list; }; 1.定义:LIST_HEAD(name)语句即可定义一个变量名为name的struct list_head结构体。 其实现机制为: 在list.h中定义了一个便于定义该结构体的宏: #define LIST_HEAD_INIT(name) {(name), (name)} #define LIST_HEAD(name) \ ??? struct list_head name = LIST_HEAD_INIT(name) 如上所示,LIST_HEAD(name)即为struct list_head的定义,变量名为name,并且使 name.next = name, name.prev = name,即为一个指向自己的空的双向循环连表。 该宏在定义struct list_head结构体时,并对其进行了初始化。 2.初始化:对于双向连表的初始化,list.h中定义了两种方法,一种是宏定义,一种是用函数实现 宏定义: 在list.h中定义了一个用来初始化struct list_head结构的宏,其实现如下所示: #define LIST_HEAD_INIT(name) {(name), (name)} 该初始化方式即上述LIST_HEAD(name)中用到的初始化方法,即将next和prev这两栏的值设为跟结构体的地址相同。 函数定义: static inline void INIT_LIST_HEAD(struct list_head *list) { ??? list-next = list; ??? list-prev = list; } 该函数与上述的宏定义的实现机制相同,只是将其翻译为了函数形式。 3.遍历: 4.删除: static inline void __list_del(struct list_head *prev, struct list_head *next) { next-prev = prev; prev-next = next; } 该方法的参数解析:第一个参数为struct list_head类型为要删除的节点的前一个元素即prev指针所指向的内容,第二个参数也为struct list_head类型的,它是要删除节点的下一个元素,即next指针指向的内容。 该方法的基本思想是:使一个双链表的前一个指针prev指向它的下一个元素,而下一个指针next则指向其前一个元素,这样该节点就被架空了,即被从链表中删除了。 不安全的删除方法: ?? ?static inline void list_del(struct list_head *entry) ?? ?{ ?? ??? ?__list_head(entry-prev, entry-next); ?? ??? ?entry-next = LIST_POISON1; ?? ??? ?entry-prev = LIST_POISON2; ?? ?} 该方法调用第一个函数,将entry从链表中删除,在删除之后为确保该节点的两个指针乱指,影响到其他数据。因此要对其两个指针进行处理。在该函数中为这两个指针赋值为LIST_POISON1和LIST_POISON2,这两个是宏定义的常量,在include/linux/poison.h中定义的,其宏定义如下所示: #define LIST_POISON1? ((void *) 0+ POISON_POINTER_DELTA) #define LIST_POISON2? ((void *) 0+ POISON_POINTER_DELTA) 这两个值是两个用户不可用的地址,就相当于把prev和next屏蔽掉,不能通过它在切入到链表中,但是它属于不安全的方式。 安全的删除方法: ?? ?static inline void list_del_init(struct list_head *entry) ?? ?{ ?? ??? ?__list_del(entry-prev, entry-next); ?? ??? ?INIT_LIST_HEAD(entry); ?

文档评论(0)

xcs88858 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

版权声明书
用户编号:8130065136000003

1亿VIP精品文档

相关文档