/******************** * 内核中链表的应用 ********************/ (1)介绍 在Linux内核中使用了大量的链表结构来组织数据,包括设备列表以及各种功能模块中的数据组织...这些链表大多采用在include/linux/list.h实现的一个相当精彩的链表数据结构。...和以前介绍的双链表结构模型不同,这里的list_head没有数据域。在Linux内核链表中,不是在链表结构中包含数据,而是在数据结构中包含链表节点。...如: struct my_struct{ struct list_head list; unsigned long dog; void *cat; }; linux中的链表没有固定的表头,从任何元素开始访问都可以...定义在 a.增加节点 list_add(struct list_head *new, struct list_head *head); 向指定链表的head
引言 链表的实现是基于结构体与指针两者实现的,常用的链表数据结构如下: //将int起别名ELEMTYPE,是为了方便修改链表中的数据域类型。...在Linux中设计了一种适合于各种类型数据域都可以使用的通用型链表: struct list_head { struct list_head *prev, *next; }; 摒弃掉数据域,只保留头尾指针...内核链表 链表的主要意义就是将分散地址的数据域通过指针排列成有序的队列。因此数据域是链表不可或缺的一部分,但是在实际使用中需要不同类型的数据域,因此也就限制了链表的通用。...Linux中在声明中抛弃了数据域,也就解决掉了这一问题。 原理 Linux使用链表的方法:使用时,自定义结构体包含数据域+链表结构体。...链表.png 如上图所示,将结构体A、B、C中的内核结构体指针构建成双链表,这样结构体A、B、C中的链表成员就可以互相索引了。
链表是基本数据结构, 一开始学习数据结构时, 我一般这么定义, 对应实现从头或尾插入的处理函数, struct int_node_old { int val; struct int_node_old...想起前段时间, 看到FreeRTOS提供的链表处理方式(《 FreeRTOS 任务调度 List 组织 》), 将链表结构定义和实际使用时具体节点数据内容分开定义, 供系统各个模块使用。...查看linux的源码, 发现linux中也为我们提供了相似的实现(源码), 把一些共性统一起来。 类是 python 中for_each处理,有些意思。...linux 下的链表定义在文件 include/linux/types.h, 采用的是双向列表 struct list_head { struct list_head *next, *prev;...list 利用这个定义, 我定义了一个自己的list数据结构, 并copy了一些接口实现,感受下,linux 是如何管理链表的。
linux kernel中的list估计已经被各位前辈们写烂了,但是我还是想在这里记录一下; linux kernel里的很多数据结构都很经典, list链表就是其中之一 本篇要介绍的内容: list...的定义 list提供的操作方法 注意事项 使用实例 ---- List 所在文件: List的所有操作可以在 include/linux/list.h找到; List head的定义可以在 include.../linux/types.h找到; 定义 实际上这就是一个双向循环链表, 且有一个头指针 list head的定义: struct list_head { struct list_head *next...__list_del_entry(list); // 添加到新链表的头部 list_add(list, head); } 将一个元素移动到另一个list的队尾 static...struct中,这个宏就是由这个list_head ptr来获取当前所处的struct对象的指针, 用了linux的经典宏定义 container_of #define list_entry(ptr,
概要 本文对双向链表进行探讨,介绍的内容是Linux内核中双向链表的经典实现和用法。其中,也会涉及到Linux内核中非常常用的两个经典宏定义offsetof和container_of。...内容包括: 1.Linux中的两个经典宏定义 2.Linux中双向链表的经典实现 Linux中的两个经典宏定义 倘若你查看过Linux Kernel的源码,那么你对 offsetof 和 container_of...Linux中双向链表的经典实现 1.Linux中双向链表介绍 Linux双向链表的定义主要涉及到两个文件: include/linux/types.h include/linux/list.h Linux...list_del(entry) 和 list_del_init(entry)是linux内核的对外接口。 list_del(entry) 的作用是从双链表中删除entry节点。...3.Linux中双向链表的使用示例 双向链表代码(list.h): 1 #ifndef _LIST_HEAD_H 2 #define _LIST_HEAD_H 3 // 双向链表节点 4 struct
printf("num = %d, math = %d\n", temp->num, temp->math); } printf("\n"); return 0; } 运行效果: 内核双链表效果图...: 大体的效果图就是如此,增加一个节点,删除一个节点都是基于这个模型展开的。...读者可以手动画画增加和删除的操作。 其实关于内核中链表的操作还有很多的函数,目前就分析这几个。其余留给自己尝试。
描述 在linux内核中封装了一个通用的双向链表库,这个通用的链表库有很好的扩展性和封装性,它给我们提供了一个固定的指针域结构体,我们在使用的时候,只需要在我们定义的数据域结构体中包含这个指针域结构体就可以了...传统的链表结构 struct node{ int key; int val; node* prev; node* next; } linux 内核通用链表库结构 提供给我们的指针域结构体...list_add-先入先出模式 我们的链表节点,实际在内存中的展示形态 ?...反推结构体首地址 举个例子 这个例子包括简单的增、删、遍历 #include #include #include <linux...内核提供的这个通用链表库里面还有很多其他的接口,这里没有详细的一一举例,有兴趣的可以自己去看看,在源码包 include/linux/list.h 文件里面,不过通过阅读一些源代码确实对我们也有很大的提高
前言: 在上期文章中,已经给大家分享过offsetof()和container_of两个宏函数,这两个宏函数在Linux内核链表里面有大量的应用,对于我们平时工作写代码有很大的帮助。...下面是Linux内核链表的内容分享。...做内核驱动开发经常会使用linux内核最经典的双向链表 list_head, 以及它的拓展接口(或者宏定义): list_add , list_add_tail, list_del , list_entry...; }; 然后就开始围绕这个结构开始构建链表,然后插入、删除节点 ,遍历整个链表等等,其实内核已经提供好了现成的接口,接下来就让我们进入 kernel/include/linux/list.h中: 一...做linux驱动开发的同学是不是想到了LDD3这本书中经常使用的一个非常经典的宏定义呢!
*p) { wait->next = wait; *p = wait; } else { /* 在第一个节点后面插入节点,形成单向循环链表 thanks to...,插入第二个节点的时候, 第一个节点的下一个节点是自己。...wait->next即新节点的next指向第一个节点的下一个节点, (*p)->next = wait;即第一个节点的next指针指向新加入的节点。...传统的头插法只能形成单链表,不能循环,因为循环需要拿尾指针的next指向第一个 节点,但是随着链表的变成,无法找到尾节点。...wait->next = wait; *p = wait; } else { // 头插法,形成单向链表
简介 链表是Linux 内核中最简单,最普通的数据结构。...链表是一种存放和操作可变数量元素(常称为节点) 的数据结构,链表和静态数组的不同之处在于,它所包含的元素都是动态创建并插入链表的,在编译 时不必知道具体需要创建多少个元素,另外也因为链表中每个元素的创建时间各不相同...根据它的特性,链表可分为:单链表,双链表,单向循环链表和双向循环链表,今天总结记录的就是 最简单的单链表, 1.1 节点类型描述 1 typedef struct node_t { ...链表基本运算的相关"算法"操作 or 操刀(~烹羊宰牛且为乐,会须一饮三百杯~) 链表的运算除了上面的创建空链表,还有数据的插入,删除,查找等函数,链表的运算有各种实现方 法,如何写出一个高效的...,封装性较好的函数是我们要考虑的,比如数据插入函数,我们就要尽可能 考虑所有能出现的结果,比如:1)如果需插入数据的链表是个空表;2)所插入的位置超过了链表的 长度;如果我们的函数能包含所有能出现的情况
【Leetcode21】合并两个有序链表 1.链接 合并两个有序链表 2.题目再现 3.三指针尾插法 思路:创建一个新的链表,分别遍历两个链表,小的就尾插到新链表,然后指针向后走一步,直到有一方为空时就结束循环...;结束循环后,判断哪个链表不为空,把不为空的尾插到新链表中去。...【Leetcode160】相交链表 1.链接 相交链表 2.题目再现 3.解法 1.先分别遍历两个链表,记录下两个链表的长度; 2.如果两个链表尾节点的地址一样,则说明它们相交,否则不相交,(注意是地址不是值...); 3.求出两个链表长度的差gap; 4.先让长的链表走差距步gap,短的链表先不动; 5.然后两个链表同时走一步,比较每走一步时两个链表当前节点的地址,如果一样,则说明找到了它们相交的起始位置...1.找到链表的中间节点; 2.逆置链表中间节点以后的部分,rmid 为后半部分逆置后的第一个节点; 3.头指针 head 和 rmid 同时向后遍历,若 head 的值不等于 rmid 的值,则不是回文结构
前言 ---- 链表中的数据通过指针连接,添加、插入或删除节点只需要修改指针指向 实现思路 实现一个链表需要具备以下方法 在链表尾部添加节点 获取链表所有节点的数据 链表指定位置插入元素 获取链表指定位置的节点数据...获取节点在链表中的位置 更新链表指定位置的数据 移除链表指定位置的节点 移除链表中的指定节点 判断链表是否为空 获取链表长度 链表内部需要定义head指针和链表长度 实现代码 定义head指针和length...this.length += 1 } 获取链表所有节点的数据 toString() { //声明空字符串存储链表节点数据 let listString = '' //获取第一个节点...(linkedList.size()) 双向链表 双向链表的指针是双向的,前指针指向上一个节点,后指针指向下一个节点 head指向第一个节点,tail指向最后一个节点 双向链表实现思路 需要具备以下方法...尾部插入元素 任意位置插入元素 获取所有节点数据 正向遍历链表获取节点数据 反向遍历链表获取节点数据 获取指定位置的节点数据 获取指定数据在链表中的位置 更新指定位置的节点数据 移除指定位置的节点 移除指定数据的节点
链表是一种常见的基础数据结构,结构体指针在这里得到了充分的利用。...链表可以动态的进行存储分配,也就是说,链表是一个功能极为强大的数组,他可以在节点中定义多种数据类型,还可以根据需要随意增添,删除,插入节点。链表都有一个头指针,一般以head来表示,存放的是一个地址。...链表中的节点分为两类,头结点和一般节点,头结点是没有数据域的。链表中每个节点都分为两部分,一个数据域,一个是指针域。...作为有强大功能的链表,对他的操作当然有许多,比如:链表的创建,修改,删除,插入,输出,排序,反序,清空链表的元素,求链表的长度等等。 ... 删除链表节点 删除链表的元素也就是把前节点的指针域越过要删除的节点指向下下个节点。
链表是一种常见的基础数据结构,结构体指针在这里得到了充分的利用。...链表可以动态的进行存储分配,也就是说,链表是一个功能极为强大的数组,他可以在节点中定义多种数据类型,还可以根据需要随意增添,删除,插入节点。链表都有一个头指针,一般以head来表示,存放的是一个地址。...链表中的节点分为两类,头结点和一般节点,头结点是没有数据域的。链表中每个节点都分为两部分,一个数据域,一个是指针域。...作为有强大功能的链表,对他的操作当然有许多,比如:链表的创建,修改,删除,插入,输出,排序,反序,清空链表的元素,求链表的长度等等。 ...下面是一个传入链表和要修改的节点,来修改值的函数。
链表的基本操作 单链表 链表的基本操作 一:单链表的基础操作 二:单链表的建立 头插法 尾插法 三:单链表的遍历 四:单链表结点数目判断 五:单链表的插入 链表头插入 任意结点插入 链表尾部插入...六:单链表的删除 七 :单链表的查询 一:单链表的基础操作 为什么需要链表?...---- 二:单链表的建立 单链表的建立即从无到有创建一个链表,一个一个的分配结点的储存空间,然后输出每一个结点的数据域,然后建立结点之间的关系。...单链表的建立可以分为两种方法,(1)头插法,(2)尾插法(更易理解) 头插法 即在单链表的头部插入新的结点的方法成为头插法。 数据读入顺序和链表的结点顺序正好相反。...计数器,每移动一次p指针且p指向不为空,iCount++; ---- 五:单链表的插入 链表的插入,有三种方式,可以从链表的头部插入,可以从链表的尾部插入,也可以在指定位置进行插入。
链表概念: 链表使用说明: 画图示意: 静态链表 #define _CRT_SECURE_NO_WARNINGS #include typedef struct LinkNode...LinkNode* next; }Lk,*lk; //Lk ------>struct LinkNode //lk----->struct LinkNode* void test01() { //静态链表..., nodeCurrent->num); nodeCurrent = nodeCurrent->next; } } int main() { test01(); return 0; } 动态链表...LinkNode* next; }Lk,*lk; //Lk ------>struct LinkNode //lk----->struct LinkNode* void test01() { //动态链表
在 Linux 内核中使用最多的数据结构就是链表了,其中就包含了许多高级思想。 比如面向对象、类似C++模板的实现、堆和栈的实现。 1....如果去掉前驱指针,就是单循环链表。 ? 2. 内核链表 在Linux内核中使用了大量的链表结构来组织数据,包括设备列表以及各种功能模块中的数据组织。...这些链表大多采用在[include/linux/list.h]实现的一个相当精彩的链表数据结构。事实上,内核链表就是采用双循环链表机制。 内核链表有别于传统链表就在节点本身不包含数据域,只包含指针域。...当 list1 被挂接到 list2 之后,作为原表头指针的 list1 的next、prev仍然指向原来的节点,为了避免引起混乱,Linux提供了一个list_splice_init()函数.该函数在将...总结 本文详细分析了 linux 内核 中的双链表结构,以图文的方式旨在帮助大家理解。
序 本文主要记录一下leetcode链表之删除链表的节点 题目 给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点。 返回删除后的链表的头节点。...注意:此题对比原题有改动 示例 1: 输入: head = [4,5,1,9], val = 5 输出: [4,1,9] 解释: 给定你链表中值为 5 的第二个节点,那么在调用了你的函数之后,该链表应变为...示例 2: 输入: head = [4,5,1,9], val = 1 输出: [4,5,9] 解释: 给定你链表中值为 1 的第三个节点,那么在调用了你的函数之后,该链表应变为 4 -> 5 ->...说明: 题目保证链表中节点的值互不相同 若使用 C 或 C++ 语言,你不需要 free 或 delete 被删除的节点 来源:力扣(LeetCode) 链接:https://leetcode-cn.com...cursor.next; } preNode.next = preNode.next.next; return head; } } 小结 这里的关键在于要设计一个
序 本文主要记录一下leetcode链表之删除链表的节点 OIP (45).jpeg 题目 给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点。 ...返回删除后的链表的头节点。 ...注意:此题对比原题有改动 示例 1: 输入: head = [4,5,1,9], val = 5 输出: [4,1,9] 解释: 给定你链表中值为 5 的第二个节点,那么在调用了你的函数之后,该链表应变为...示例 2: 输入: head = [4,5,1,9], val = 1 输出: [4,5,9] 解释: 给定你链表中值为 1 的第三个节点,那么在调用了你的函数之后,该链表应变为 4 -> 5 ->...说明: 题目保证链表中节点的值互不相同 若使用 C 或 C++ 语言,你不需要 free 或 delete 被删除的节点 来源:力扣(LeetCode) 链接:https://leetcode-cn.com
领取专属 10元无门槛券
手把手带您无忧上云