数据结构笔记-双向链表

双向循环链表:
双向: 链表中任意一个节点有2个指针,一个指针指向前驱节点,一个指针指向后驱节点
循环: 链表中的末尾节点的后驱节点是 head 头结点
head头结点的前驱节点是 链表中的末尾节点

struct dlist{
//数据域
struct dlist *next;//指向后驱节点
struct dlist *prev;//指向前驱节点
};

双向循环链表:
(1)创建头结点
(2)创建普通节点
(3)增删查改
(4)遍历


双向循环链表的优化。
(1)把增加节点单独封装,封装成添加到任意两个相邻节点之间
(2)把遍历时的while循环封装成for循环的宏定义

因为在linux内核中,存在一个内核链表,它编程思想就是通过宏来管理链表

grep “struct list_head {” -r /usr
grep 查找某一个文件是否包含某个字符串
“字符串” 查找的目标
-r 递归查找,也就是查找多个文件
/usr 查找的目标文件夹


内核链表
(1)设置好链表的API接口,可以直接调用
(2)把指针域封装成结构体,通过指针域结构体的链式连接,
可以串联起不同数据类型的大结构体。

当需要进行增删查改,遍历的时候,都是通过小结构体执行的。

为了修改或者获取大结构体中数据域的值,需要通过小结构体找到大结构体的函数。

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

#define list_entry(ptr, type, member)
container_of(ptr, type, member)

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
====》合并

list_entry(ptr, type, member)
{
const typeof( ((type *)0)->member ) *__mptr = (ptr);
(type *)( (char *)__mptr - ((size_t) &((TYPE *)0)->MEMBER));
}

//这个for循环中不能删除节点,因为它在每次循环结束以后,要再一次用到pos指针
//如果循环时删除掉了pos指针指向的节点,那么循环结束的执行语句 pos->member.next 会报错。
#define list_for_each_entry(pos, head, member)
for (pos = list_entry((head)->next, typeof(*pos), member);
&pos->member != (head);
pos = list_entry(pos->member.next, typeof(*pos), member))

所以当需要在循环过程中删除节点的时候,必须用
#define list_for_each_entry_safe(pos, n, head, member)
for (pos = list_entry((head)->next, typeof(*pos), member),n = list_entry(pos->member.next, typeof(*pos), member);
&pos->member != (head);
pos = n, n = list_entry(n->member.next, typeof(*n), member))
在这个宏定义中,pos在每次循环结束后重新赋值了,所以不会出现 pos删除后 报错的情况

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值