【脚踢数据结构】内核链表

这篇具有很好参考价值的文章主要介绍了【脚踢数据结构】内核链表。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

  • (꒪ꇴ꒪ ),Hello我是祐言QAQ
  • 我的博客主页:C/C++语言,Linux基础,ARM开发板,软件配置等领域博主🌍
  • 快上🚘,一起学习,让我们成为一个强大的攻城狮!
  • 送给自己和读者的一句鸡汤🤔:集中起来的意志可以击穿顽石!
  • 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏

        内核链表(Kernel Linked List)是操作系统内核中常用的一种数据结构,用于管理和维护一系列数据元素(节点)。它也是一种线性数据结构,其中每个节点包含了数据元素本身以及指向下一个节点的指针。内核链表在操作系统中广泛应用于管理进程、文件描述符、内存分配等诸多场景。

一、内核链表概述

        内核链表通常由一个特定的数据结构定义,该数据结构包含一个或多个指向链表中首个和最后一个节点的指针,以及其他用于操作和管理链表的属性。在C语言中,内核链表的定义示例:

//数据
typedef int Datatype;

//节点(大结构体)
typedef struct Kernel_node
{
	Datatype data;			//数据域
	struct kernel_list list;	//指针域
}k_node;

//指针节点(小结构体)
struct kernel_list {
    struct kernel_list *prev;//前驱指针
    struct kernel_list *next;//后继指针
};


        kernel_list结构体定义了一个内核链表的节点,其中prev指向前一个节点,next指向下一个节点,这一点在某种意义上与常规的双向循环链表一致,但是还是有区别的。我们再来看看之前所学习的双向循环链表的节点定义:

//数据
typedef int Datatype;

//节点
typedef struct Node
{
	DataType data;		//数据域
	struct Node *prev;	//指针域:前驱指针
	struct Node *next;	//指针域:后继指针
}node;

        这样一对比,就会发现常规链表与内核链表最大的区别就是嵌套,内核链表的节点结构通常嵌套在某个容器数据结构中,可以理解为能单独对指针域进行操作,而不影响数据。

【脚踢数据结构】内核链表,脚踢数据结构,数据结构,链表,算法,linux

 所以这也暴露了常规链表的缺陷:

        (1)每一种应用中,节点都是特殊的,导致每一条链表都是特殊的,因此每一种链表的增删查改这些算法也都是特殊的;

        (2)当一个节点处于变化的数据结构网络中的时候,节点指针无法指向稳定不变的节点。

        这样的描述或许不能让你有直观的感觉,但当你往下看,真正去理解了内核链表再回过头来,你就会豁然开朗。

二、内核链表的原理

        内核链表的节点结构通常嵌套在某个容器数据结构中,以实现紧密关联。节点结构中至少包含两个指针,一个指向前一个节点,一个指向后一个节点。这样的设计使得在链表中插入、删除节点的操作更加高效,无需像数组那样移动大量元素。我们可以这样理解:

        (1)将链表的结构抽象出来,去除节点中的具体数据,只保留逻辑的双向指针,形成一条只包含逻辑的“纯粹的链表”。

        (2)将链表“寄宿”于具体的数据节点之中,使他贯穿这些节点,可以借助一定的方式通过“纯粹链表“的指针域得到数据节点。

【脚踢数据结构】内核链表,脚踢数据结构,数据结构,链表,算法,linux

            内核链表做到了将数据和逻辑(指针)分开,在红色方框内的这样只有前后两指针的链表形式被称为Linux标准双向循环链表

三、内核链表的操作

        在学习内核链表的操作之前,我们先要来了解一下list.h文件。list.h 是一个常见的头文件,通常用于实现内核链表(双向链表)的相关功能。在 Linux 内核中,list.h 提供了一种高效的方式来操作链表,包括插入、删除、遍历等操作。完整的该文件可从网上各处找到,接下来我们将只举例用到的结构体。

1. 节点声明

//数据
typedef int Datatype;

//链表节点(大结构体)
typedef struct Kernel_node
{
	Datatype data;			//数据域
	struct list_head list;	//指针域
}k_node;

         list_head 是在list.h中已经定义好的指针域部分,我们可以将其称之为小结构体,用它来构建“纯粹链表”。


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

2.初始化链表

//初始化链表
k_node *init_kernel_list()
{
	k_node *head = malloc(sizeof(k_node));
	if (head==NULL)
	{
		perror("malloc");
		exit(0);
	}
	else
	{		
		INIT_LIST_HEAD(&(head->list));//带参宏,初始化小结构体里面的这两个指针成员

	}
	return head;
}

        INIT_LIST_HEAD也是list.h中已经定义好的带参宏,原型如下:

#define INIT_LIST_HEAD(ptr) do { \
	(ptr)->next = (ptr); (ptr)->prev = (ptr); \
} while (0)

3.创建节点

//创建节点
k_node *create_kernel_node(Datatype data)
{
	k_node *new = malloc(sizeof(k_node));
	if (new != NULL)
	{
		new->data = data;
		new->list.next = NULL;
		new->list.prev = NULL;
	}
	return new;
}

4.遍历链表


//遍历链表
void display(k_node *head)
{
	struct list_head *pos = NULL;
	k_node *Node = NULL;
	list_for_each(pos, &(head->list))
	{
		Node = list_entry(pos, k_node, list);
		printf("%d ", Node->data);
	}
	printf("\n");
}

        其中list_for_each和list_entry也是在list.h中定义好的带参宏:


//遍历链表的for循环,每一次循环体内得到的就是一个小结构体指针
#define list_for_each(pos, head) \
for (pos = (head)->next; pos != (head); \
pos = pos->next)


//小结构体指针ptr,大结构体类型type,小结构体在大结构体内部的成员名
#define list_entry(ptr, type, member) \
((type *)((char *)(ptr)-(unsigned long)(&((type *)0)->member)))

        做完这些操作,我们就可以通过一个实例来验证我们的操作是否正确,我们可以创建一个10个节点的链表并打印。

【脚踢数据结构】内核链表,脚踢数据结构,数据结构,链表,算法,linux

         实例代码如下:

#include <stdio.h>
#include <stdlib.h>
#include <stdbool.h>
#include "list.h"

//数据
typedef int Datatype;

//链表节点
typedef struct Kernel_node
{
	Datatype data;			//数据域
	struct list_head list;	//指针域
}k_node;

//初始化链表
k_node *init_kernel_list()
{
	k_node *head = malloc(sizeof(k_node));
	if (head==NULL)
	{
		perror("malloc");
		exit(0);
	}
	else
	{
		INIT_LIST_HEAD(&(head->list));
	}
	return head;
}

//创建节点
k_node *create_kernel_node(Datatype data)
{
	k_node *new = malloc(sizeof(k_node));
	if (new != NULL)
	{
		new->data = data;
		new->list.next = NULL;
		new->list.prev = NULL;
	}
	return new;
}

//遍历链表
void display(k_node *head)
{
	struct list_head *pos = NULL;
	k_node *Node = NULL;
	list_for_each(pos, &(head->list))
	{
		Node = list_entry(pos, k_node, list);
		printf("%d ", Node->data);
	}
	printf("\n");
}

int main(int argc, char const *argv[])
{
	k_node *head = init_kernel_list();
	for (int i = 0; i < 10; ++i)
	{
		k_node *new = create_kernel_node(i+1);

		//插入节点到链表
		list_add_tail(&(new->list), &(head->list));
	}

	//遍历链表
	display(head);

	return 0;
}

        更安全的遍历列表方法补充:

void display2(k_node *head)
{
	struct list_head *pos = NULL;
	struct list_head *n = NULL;//为了防止你在遍历的时候,删除节点,找不到下一个节点的地址
	k_node *Node;
	list_for_each_safe(pos, n, &(head->list))
	{
		Node = list_entry(pos, k_node, list);
		printf("%d ", Node->data);
	}
	printf("\n");
}

void display3(k_node *head)
{
	k_node *pos = NULL;
	list_for_each_entry(pos, &(head->list), list)
	{
		printf("%d ", pos->data);
	}
	printf("\n");
}

void display4(k_node *head)
{
	k_node *pos = NULL;
	k_node *n = NULL;
	list_for_each_entry_safe(pos, n, &(head->list), list)
	{
		printf("%d ", pos->data);
	}
	printf("\n");
}

5.删除节点

void delete_node(k_node *head, Datatype data)
{
	struct list_head *pos = NULL;
	struct list_head *n = NULL;//为了防止你在遍历的时候,删除节点,找不到下一个节点的地址
	k_node *Node = NULL;
	list_for_each_safe(pos, n, &(head->list))
	{
		Node = list_entry(pos, k_node, list);
		if (Node->data == data)
		{
			list_del(&(Node->list));
		}		
	}
}

6.移动节点

void move_node_head(k_node *head, Datatype d1, Datatype d2)
{
	k_node *p1 = find_node(head, d1);
	k_node *p2 = find_node(head, d2);
	if (p1==NULL || p2==NULL)
	{
		return;
	}
	list_move(&(p1->list), &(p2->list));
}

void move_node_tail(k_node *head, Datatype d1, Datatype d2)
{
	k_node *p1 = find_node(head, d1);
	k_node *p2 = find_node(head, d2);
	if (p1==NULL || p2==NULL)
	{
		return;
	}
	list_move_tail(&(p1->list), &(p2->list));//尾插
}
	//移动节点
	move_node_head(head, 1, 7);
	display4(head);

【脚踢数据结构】内核链表,脚踢数据结构,数据结构,链表,算法,linux

7.判断是否为空链表

k_node *list_empty(k_node *head)
{
	return head->next == head;
}

8.合并链表

int main(int argc, char const *argv[])
{
	k_node *head = init_kernel_list();
	for (int i = 0; i < 10; ++i)
	{
		k_node *new = create_kernel_node(i+1);

		//插入节点到链表
		list_add_tail(&(new->list), &(head->list));//尾插
	}
	display4(head);

	k_node *head1 = init_kernel_list();
	for (int i = 10; i < 15; ++i)
	{
		k_node *new = create_kernel_node(i+1);

		// 插入节点到链表
		list_add_tail(&(new->list), &(head1->list));//尾插
	}
	display4(head1);

	if (list_empty(&(head1->list)))
	{
		printf("这是空链表\n");
	}
	else
	{
		printf("非空,可合并\n");
	}
	// head的头节点下面的第一个节点的前驱指针会断开,head的尾节点的后继指针会断开,head后面不能在使用
	list_splice_init(&(head->list), &(head1->list));

	display4(head1);

	return 0;
}

        运行结果:

【脚踢数据结构】内核链表,脚踢数据结构,数据结构,链表,算法,linux

 

留个作业吧

        用内核链表创建一个数据集合,初始包含数据:1 2 3 4 5 6 7 8 9 ,将其重新排列成:9 7 5 3 1 2 4 6 8 (奇数降序,偶数升序)并显示出来。

        答案评论区见!

四、总结

        与一般链表相比,内核链表有一些特点和优势:

        1.容器结构体: 在内核链表中,节点通常嵌套在某个数据结构中,这个数据结构即所谓的容器结构体。这种设计可以让数据元素和链表节点紧密关联,方便数据的操作和管理。

        2.稳定性和预测性: 内核链表在操作系统内核中广泛应用,而且内核链表的实现通常被优化为在各种情况下都能够稳定和预测地工作,不容易受到异常情况的影响。

        3.特定操作: 内核链表通常提供一系列针对特定操作的函数,如插入、删除、遍历等。这些函数经过精心的设计和优化,能够高效地处理链表操作。

        总之,内核链表是操作系统内核中的一种重要数据结构,用于高效地管理各种资源和数据。它与一般链表相比,更注重稳定性和性能优化,以满足操作系统的特定需求。

        更多C语言Linux系统ARM板实战数据结构相关文章,关注专栏:

   手撕C语言

            玩转linux

                    脚踢数据结构

                            6818(ARM)开发板实战文章来源地址https://www.toymoban.com/news/detail-637404.html

📢写在最后

  • 今天的分享就到这啦~
  • 觉得博主写的还不错的烦劳 一键三连喔~
  • 🎉感谢关注🎉

到了这里,关于【脚踢数据结构】内核链表的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【脚踢数据结构】深入理解栈

    (꒪ꇴ꒪ ),Hello我是 祐言QAQ 我的博客主页:C/C++语言,Linux基础,ARM开发板,软件配置等领域博主🌍 快上🚘,一起学习,让我们成为一个强大的攻城狮! 送给自己和读者的一句鸡汤🤔: 集中起来的意志可以击穿顽石! 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏

    2024年02月13日
    浏览(44)
  • 【脚踢数据结构】队列(顺序和链式)

    (꒪ꇴ꒪ ),Hello我是 祐言QAQ 我的博客主页:C/C++语言,Linux基础,ARM开发板,软件配置等领域博主🌍 快上🚘,一起学习,让我们成为一个强大的攻城狮! 送给自己和读者的一句鸡汤🤔: 集中起来的意志可以击穿顽石! 作者水平很有限,如果发现错误,可在评论区指正,感谢🙏

    2024年02月12日
    浏览(38)
  • Linux内核数据结构 散列表

    在Linux内核中,散列表(哈希表)使用非常广泛。本文将对其数据结构和核心函数进行分析。和散列表相关的数据结构有两个: hlist_head 和 hlist_node 对应的结构如下 hash_table 为散列表(数组),其中的元素类型为 struct hlist_head 。以 hlist_head 为链表头的链表,其中的节点 hash 值

    2024年02月10日
    浏览(40)
  • 【算法与数据结构】链表

    链表是由一串节点串联在一起的,链表的每个节点存储两个信息:数据+下一个节点的地址 分清楚两个概念:什么是内存内部,什么是程序内部 内存内部: 信息存储在内存空间里的 程序内部: 通过什么信息,去操作结构 如果想操作链表的话,我们依靠的是程序内部的信息,

    2024年02月03日
    浏览(45)
  • 【数据结构与算法】链表

    对于顺序表存在一些缺陷: 中间/头部的插入删除,时间复杂度为O(N) 。头部插入需要挪动后面的元素 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们再继续插

    2023年04月09日
    浏览(48)
  • 02-链表 (数据结构和算法)

    3.1 链表的基本概念 前面我们在学习顺序表时,线性表的顺序存储结构的特点是逻辑关系上相邻的两个数据元素在物理位置上也是相邻的。我们会发现虽然顺序表的查询很快,时间复杂度为O(1),但是增删的效率是比较低的,因为每一次增删操作都伴随着大量的数据元素移动。为

    2024年02月16日
    浏览(46)
  • Python数据结构与算法-数据结构(列表、栈、队列、链表)

    数据结构是指相互之间存在这一种或者多种关系的数据元素的集合和该集合中元素之间的关系组成。 简单来说,数据结构就是设计数据以何种方式组织并存储在计算机中。 比如:列表、集合与字典等都是一种数据结构。 N.Wirth:“程序=数据结构+算法” 数据结构按照其 逻辑结

    2024年02月08日
    浏览(58)
  • Linux内核代码中常用的数据结构

    Linux内核代码中广泛使用了数据结构和算法,其中最常用的两个是链表和红黑树。 Linux内核代码大量使用了链表这种数据结构。链表是在解决数组不能动态扩展这个缺陷而产生的一种数据结构。链表所包含的元素可以动态创建并插入和删除。 链表的每个元素都是离散存放的,

    2024年02月12日
    浏览(32)
  • 数据结构与算法:双向链表

    朋友们大家好啊,在上节完成单链表的讲解后,我们本篇文章来对 带头循环双向链表进行讲解 单链表中,一个节点存储数据和指向下一个节点的指针,而双向链表除了上述两个内容,还包括了 指向上一个节点的指针 带头的双向链表,是指在双向链表的最前端添加了一个 额

    2024年02月20日
    浏览(51)
  • 【数据结构与算法】双向链表

    作者:旧梦拾遗186 专栏:数据结构成长日记   带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。 现在我们来通

    2024年02月11日
    浏览(57)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包