list_head结构体的理解

本文介绍了双向链表的基本概念及list_head结构体的应用。详细解释了如何利用list_head构建双向链表,并通过实例演示了链表的创建、节点添加等操作。

1、 list_head之前(链表归整)

1.1、 单向链表

原型:
struct simplex
{
	Int value;
<span style="white-space:pre">	</span>struct simplex *next;
};
结构图如下所示


特点是单向只能从前往后查找,橙色线代表单向循环链表相较于单向链表所多出来的线

1.2、 双向链表

原型:
struct bothlist
{
	struct bothlist *next;
	struct bothlist *prev;
};


特点是双向,从前往后,从后往前都可以查询,橙色线是双向循环链表相较双向链表所多出来的两条线


2、 list_head结构体

2.1、简介

原型:
struct list_head {
	struct list_head *next, *prev;
};

此结构体所构成的链表如上面的双向循环链表图示
此结构体在linux内核中被大量的引用,几乎所有内核当中需要构成链表结构的地方都用到了这个结构体。例如内核的总线设备就用到了这个结构体

2.2、此结构体的作用

例如现在有任意给定的一个结构体需要组成一个双向链表,则可以在此结构体当中加入list_head结构体用来进行链表的构建,例如结构体原型如下
struct test_list
{
	unsigned char *name;
	struct list_head test;
	int value;
};

下面是根据此结构体以及list_head文件中的函数编写的测试文件,使用GCC4.8.1进行编译
#include <stdio.h>
//#include "list.h"

struct list_head {
	struct list_head *next, *prev;
};

struct test_list
{
	unsigned char *name;
	struct list_head test;
	int value;
};

static struct test_list lists[] = {
	[0] = {
		.value = 0,
		.name = "hehe"
	},
	[1] = {
		.value = 1,
	},
	[2] = {
		.value = 2,
	},
	[3] = {
		.value = 3,
		.name = "test"
	},
	[4] = {
		.value = 4,
	}	
};

struct list_head head = {
	&(head), &(head)
};

static void __list_add_tail(struct list_head *new, struct list_head *prev, struct list_head *next)
{
	new->next = next;
	new->prev = prev;
	prev->next = new;
	next->prev = new;
}

void mlist_add_tail(struct list_head *new, struct list_head *head)
{
	__list_add_tail(new, head->prev, head);
}

#define mlist_entry(list_head, type, member_name) \
			(type *)((unsigned int)list_head - (unsigned int)(&(((type*)(0))->member_name)))


#define mlist_for_each(pos, head) \
			for(pos = (head)->prev; pos != head; pos = pos->prev)

int main(int argv, char *argc[])
{
	struct list_head *mod;
	unsigned int i = 0;
	
	mlist_add_tail(&lists[0].test, &head);
	mlist_add_tail(&lists[1].test, &head);
	mlist_add_tail(&lists[2].test, &head);
	mlist_add_tail(&lists[3].test, &head);
	mlist_add_tail(&lists[4].test, &head);
	
	mlist_for_each(mod, &head)
	{
		struct test_list *inode;
		
		inode = mlist_entry(mod, struct test_list, test);
		printf("%d %s\n", inode->value, inode->name);
	}		
		
	getchar();
	
	return 0;
}
上述代码经过编译之后运行的结果如下
4 (null)
3 test
2 (null)
1 (null)
0 hehe

后经修改
#define mlist_for_each(pos, head) \
			for(pos = (head)->prev; pos != head; pos = pos->prev)

#define mlist_for_each(pos, head) \
			for(pos = (head)->next; pos != head; pos = pos->next)
运行结果如下
0 hehe
1 (null)
2 (null)
3 test
4 (null)

由此可以得知此链表的结构为一个双向链表。

2.3、实现方法

首先是头结点的创建以及初始化
struct list_head head = {
	&(head), &(head)
};

此时head的结构体成员均指向自己,形如

链表的添加过程
static void __list_add_tail(struct list_head *new, struct list_head *prev, struct list_head *next)
{
	new->next = next;
	new->prev = prev;
	prev->next = new;
	next->prev = new;
}


由结构体内部的list_head结构体得到结构体所在的位置
#define mlist_entry(list_head, type, member_name) \
			(type *)((unsigned int)list_head - (unsigned int)(&(((type*)(0))->member_name)))

反映到代码mlist_entry(mod, struct test_list, test)得到
(struct test_list *)((unsigned int)mod - (unsigned int)(&((( struct test_list *)(0))-> test)))
分层次解析
((( struct test_list *)(0))-> test) 定义一个test_list结构体变量0,使其指向其中的test成员。
(&((( struct test_list *)(0))-> test))) 取此成员所在的地址,由于起始为0地址,所以现在获得的地址就是test_list结构体开始到其test成员的偏移值,将其强制转换为unsigned int型变量
((unsigned int)mod - (unsigned int)(&((( struct test_list *)(0))-> test)))拿取得的list_head结构体mod的位置减去偏移值得到当前mod所在的test_list结构体的起始位置,最后将这个值强制转换为test_list结构体指针,之后就得到一个test_list的地址,也就是当前list_head结构体所在的那个test_list。
其余诸如链表的删除等等操作相对容易理解,不再赘述
<think>根据之前的知识,在将新节点添加到链表之前,必须初始化该节点的`list_head`成员。这是因为链表操作宏(如`list_add`、`list_add_tail`)依赖于正确的初始化状态(即`next`和`prev`指针指向自身或有效节点)。如果不初始化,可能导致链表操作错误(如遍历时死循环或内存访问错误)。 因此,在创建新节点后,应该调用`INIT_LIST_HEAD(&new_entry->list)`来初始化其链表成员。但注意,在Linux内核链表中,`list_add`和`list_add_tail`函数内部会处理节点的链接,所以初始化是必要的,以确保在添加到链表之前节点处于独立状态(即不指向任何其他节点)。 在之前的示例中,我们使用`calloc(1, sizeof(ARP_ENTRY))`分配内存并用0填充,但这并不能保证链表指针被正确初始化(因为NULL指针并不等同于链表初始化的状态)。链表初始化宏`INIT_LIST_HEAD`会将节点的`next`和`prev`指针都指向节点自身,形成一个自环。这是链表节点未加入任何链表时的标准状态。 所以,答案是:**必须调用`INIT_LIST_HEAD()`初始化新节点的`list_head`成员**。 ### 示例代码片段 ```c // 创建新节点 ARP_ENTRY *new_entry = (ARP_ENTRY *)malloc(sizeof(ARP_ENTRY)); if (!new_entry) { // 错误处理 return; } // 初始化链表节点(必须步骤) INIT_LIST_HEAD(&new_entry->list); // 然后填充其他数据 // ... // 添加到链表尾部 list_add_tail(&new_entry->list, &head); ``` ### 为什么需要初始化? - 未初始化的链表节点指针可能是随机的,如果直接调用`list_add_tail`,虽然该函数会修改相邻节点的指针,但新节点的`next`和`prev`可能没有正确设置,导致后续遍历时出现错误(如访问非法地址)。 - 初始化后,节点处于“未链接”状态(指向自身),这样在链表操作中更安全,也便于检查节点是否在链表中(通过`list_empty`函数)。 ### 特殊情况说明 - 如果使用`list_add`或`list_add_tail`,它们会修改节点的指针,但初始化仍然必要,因为初始化确保在添加前节点处于已知状态。 - 在节点被从链表中删除后,如果需要重新加入,不需要再次初始化,因为`list_del`操作会将节点设置为未链接状态(即`next`和`prev`指向特殊值,但通常为了安全,如果节点可能被重新使用,建议再次初始化或使用`list_del_init`,它会将节点初始化为未链接状态)。 §§相关问题§§ 1. 如果不初始化`list_head`节点直接添加到链表,会导致什么具体问题? 2. 除了`INIT_LIST_HEAD`,还有哪些内核链表宏可以用于初始化节点? 3. 当从链表中删除一个节点时,`list_del`和`list_del_init`有什么区别? 4. 如何检查一个链表节点是否已经链接到某个链表? 5. 在用户空间使用Linux内核链表时,如何实现`INIT_LIST_HEAD`宏的功能?</think>### 关于 `INIT_LIST_HEAD()` 的必要性 **必须调用 `INIT_LIST_HEAD()` 初始化新节点的 `list_head` 成员!** 这是 Linux 内核链表操作的核心要求,原因如下: 1. **链表状态一致性** `INIT_LIST_HEAD()` 会将节点的 `next` 和 `prev` 指针指向自身,形成初始状态: ```c static inline void INIT_LIST_HEAD(struct list_head *list) { list->next = list; list->prev = list; } ``` 这是链表节点"未加入链表"的标准状态。 2. **避免未定义行为** 如果跳过初始化直接使用 `list_add_tail()`: ```c p_entry = calloc(1, sizeof(ARP_ENTRY)); list_add_tail(&p_entry->list, &head); // 危险! ``` 此时 `list->next/prev` 可能是随机值(calloc 清零也不能保证链表结构正确),会导致: - 链表断裂或循环引用 - 遍历时死循环或内存访问错误 - `list_empty()` 判断失效 3. **操作宏的依赖** 所有链表操作宏(如 `list_for_each_entry`)都假定节点已初始化。未初始化的节点会破坏整个链表结构。 ### 正确操作顺序 ```c // 1. 分配内存 ARP_ENTRY *new_entry = calloc(1, sizeof(ARP_ENTRY)); if (!new_entry) return; // 2. 初始化链表节点 (必须!) INIT_LIST_HEAD(&new_entry->list); // 3. 添加到链表 list_add_tail(&new_entry->list, &g_arp_param.head_node); // 4. 填充数据 new_entry->last_seen_s = time(NULL); // ... 其他赋值 ``` ### 特殊场景说明 - **`calloc` 清零不足以保证安全**: 虽然 `calloc` 会将指针置 NULL,但 `list->next = NULL` 不是合法的链表状态,会导致 `list_add_tail` 操作崩溃。 - **删除后重新加入的节点**: 如果使用 `list_del_init()` 删除节点(该宏会重置链表指针),再次加入时无需重新初始化。但新建节点必须初始化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值