struct list *next, *prev;
};
static inline void
list_init(struct list *list)
{
list->next = list;
list->prev = list;
}
static inline int
list_empty(struct list *list)
{
return list->next == list;
}
static inline void
list_insert(struct list *link, struct list *new_link)
{
new_link->prev= link->prev;
new_link->next= link;
new_link->prev->next= new_link;
new_link->next->prev= new_link;
}
注意:该操作相对比较难理解。这里把link为list节点方便理解,实际上link是任意的,该操作是把新的节点插入到link节点的前方。
list初始为空时:此时2个节点前后都是一样的节点,也就是说无所谓前后了。
list初始不为空时:
static inline void
list_append(struct list *list, struct list *new_link)
{
list_insert((struct list *)list, new_link);
}
插入到当前节点的前方,也就是说让新的节点的prev指针指向当前节点的prev指针指向的地方,让新的节点的next指针指向当前节点。
static inline void
list_prepend(struct list *list, struct list *new_link)
{
list_insert(list->next, new_link);
}
插入到当前节点的前方,也就是说让新的节点的prev指针指向当前节点,让当前节点的next指针指向新的节点。
static inline void
list_remove(struct list *link)
{
link->prev->next = link->next;
link->next->prev = link->prev;
}
#define list_entry(link, type, member) \
((type *)((char *)(link)-(unsigned long)(&((type *)0)->member)))
在实际使用中,list往往是作为某个结构体的成员,通过list的地址,推导出结构体的地址。问题又回到通过结构体中的某个成员的地址推导出该结构体的地址的问题上来了。
type:结构体类型。
member:结构体的成员。
link:结构体成员member的地址。
#define list_head(list, type, member) \
list_entry((list)->next, type, member)
通过list节点的next指向的节点的地址,推导出list节点的next指向的节点所在的结构体的地址。
#define list_tail(list, type, member) \
list_entry((list)->prev, type, member)
通过list节点的prev指向的节点的地址,推导出list节点的prev指向的节点所在的结构体的地址。
#define list_next(elm, member) \
list_entry((elm)->member.next, typeof(*elm), member)
参考以上。
#define list_for_each_entry(pos, list, member) \
for (pos = list_head(list, typeof(*pos), member);\
&pos->member != (list);\
pos = list_next(pos, member))
遍历整个list双向链表,其中参数list作为一个哨兵节点的角色,以它来作为退出的条件。
边想边写,写得太乱了,头晕了。