linux内核中的数据结构
http://vinllen.com/linuxnei-he-zhong-de-shu-ju-jie-gou/
https://zhuanlan.zhihu.com/p/58087261
https://blog.csdn.net/wenqian1991/article/details/44515713
https://blog.csdn.net/ace_an/article/details/53813242
Linux中最重要最常用如下四种:
LIST:链表 <linux/list.h>
Linux内核的标准链表就是采用“环形、双向”链表形式实现
沿着链表移动智能是线性移动
需要随机访问的数据,一般不使用链表
链表存放数据的理想情况是:需要遍历所有数据、或者需要动态加入/删除数据
有时首元素会用一个特殊的指针表示,称为“头指针”,可以方便的找到链表的“起始端”
Linux内核实现特殊性:不是将数据结构塞入链表,而是将链表结点塞入数据结构
Linux内核链表操作函数的复杂度都是O(1),而不管链表中元素数目的多少
list_add\__list_add
list_add_tail\__list_add_tail
list_del\__list_del
list_del_init\__list_del_init
list_move\__list_move
list_move_tail\__list_move_tail
list_empty\__list_empty
list_entry\__list_entry
list_splice\__list_splice
list_splice_init\__list_splice_init
Linux内核遍历链表的复杂度是O(n),n是链表元素数目
list_for_each
list_for_each_entry
list_for_each_entry_reverse 反向遍历链表,有二原因:①当反向遍历性能好时;②当遍历顺序很重要时;
list_for_each_entry_safe 该函数允许在遍历链表循环体中删除元素,普通遍历函数不允许这样做
list_for_each_entry_safe_reverse
链表操作函数分为外部函数、内部函数,函数同名,只差双下划线__,内部函数用prev、next参数
FIFO:队列 <linux/kfifo.h>
Linux内核通用队列实现称为kfifo,实现在文件kernel/kfifo.c中
维护两个偏移量:
入口偏移(下一次入队时位置)>=出口偏移(下一次出队时位置);
入口偏移==出口偏移,说明队列空了;
入口偏移==队列长度,说明队列重置前不能再有入队操作;
提供两个主要操作:
入队enqueue(copy数据到队列入口偏移位置)
出队dequeue(从队列出口偏移copy数据)
队列操作:
kfifo_alloc 动态创建、初始化 (size需是2的n次幂)
kfifo_init 动态创建、初始化,但使用已分配内存 (size同上)
DECLARE_KFIFO/INIT_KFIFO 静态声明,不常用(size同上)
kfifo_in 入队
kfifo_out 出队
kfifo_out_peek 不出队,只返回数据
kfifo_size 队列总体空间(长度)
kfifo_len 队列已用空间
kfifo_avail 队列剩余空间
kfifo_is_empty 判定队列空
kfifo_is_full 判定队列满
kfifo_reset 重置队列(抛弃所有元素)
kfifo_free 释放kfifo_alloc创建的队列
映射:
称为关联数组,由“键<==>值”对儿组成的集合;
“键”是唯一的,“值”关联到键;
映射操作:
Add (key, value) 增加
Remove (key) 删除
Value = Lookup (key) 查找
Allocate 插入键值对,产生UID
Linux内核提供了简单、有效的映射数据结构。但是它并非一个通用的映射;
Linux内核提供它的目标是:映射一个唯一的标识数(UID)到一个指针; (因此并不适合其它场景)
idr数据结构:用于映射用户空间的UID
idr_init 初始化静态/动态分配的idr
...
二叉树:
树,是一个能提供分层的“树”型数据结构的特定数据结构;
树,是一个无环、连接的有向图;
树,任何一个节点具有0~n个出边,1个入边;
节点深度,是从根节点起到达该节点止一共需要经过的父节点数目;
树的高度,是叶子节点的深度;
二叉树,任何一个节点最多有2个出边的树,即只能有0、1、2个出边;
BST:二叉搜索树
规则1:根左分支节点值 < 根节点值
规则2:根右分支节点值 > 根节点值
规则3:所有子树都是二叉搜索树
前序遍历:根节点->左子树->右子树
中序遍历:左子树->根节点->右子树
后序遍历:左子树->右子树->根节点
在树中搜索给定值、按序遍历都相当快捷;中序遍历二叉搜索树时会按照节点值从小到大的顺序排序;
自平衡二叉搜索树
平衡二叉搜索树:所有叶子节点深度差值都不超过1的二叉搜索树
自平衡二叉搜索树:操作都试图维持平衡(半平衡)的二叉搜索树
红黑树
是一种“自平衡二叉搜索树”
是Linux主要的平衡二叉搜索树
具有特殊的着色属性(或红色、或黑色)
能维持半平衡结构,其6属性:
属性1:所有节点要么着红色、要么着黑色
属性2:叶子节点都是黑色
属性3:叶子节点不包含数据
属性4:非叶子节点都有2个子节点
属性5:如果节点是红色,则它子节点都是黑色
属性6:一个节点到该节点的叶子节点的所有路径中,总是包含相同数目的黑色节点;
上述属性确保:
一个红色节点不能是其它红色节点的子节点(或者父节点),即两个红色节点不能相连;
从树的任何一个节点到其叶子节点的路径都具有相同数目的黑色节点;即最长路径是红黑交替路径,最短路径是全黑路径;
所以,最深叶子节点深度,不会大于2倍最浅叶子节点深度; 红黑树总是半平衡的;
rbtree: <linux/rbtree.h>
Linux实现的红黑树
实现在lib/rbtree.c中
插入效率和树中节点数目呈现对数关系;(即:时间复杂度与节点数目是对数关系)
根节点由数据结构rb_root描述,创建新红黑树,需要分配新的rb_root结构并初始化为RB_ROOT
其它节点由数据结构rb_node描述
未提供搜索和插入操作函数,需要由用户自己实现(可以使用rbtree提供的辅助函数,但要实现比较操作算子)
其它较少用的:
radixtree:基数树
数据结构的选择:
原则一:遍历操作为主时,优先考虑链表;(没有数据结构能提供比线性算法复杂度更好的算法去遍历元素)
原则二:排除性能因素,当需要相对较少数据项时,优先考虑链表;
原则三:当需要与其它选择链表的代码交互时,优先考虑链表;
原则四:需要大小不明的数据集合,优先选择链表;
原则五:代码架构复合"生产者/消费者"模式,优先选择队列;
原则六:当需要一个定长的缓冲,选择队列;
原则七:如果需要映射一个UID到一个对象,选择映射;
原则八:如果需要存储大量数据,并且快速检索,选择红黑树;
linux内核中的数据结构的更多相关文章
- Linux 内核中的数据结构:基数树(radix tree)
转自:https://www.cnblogs.com/wuchanming/p/3824990.html 基数(radix)树 Linux基数树(radix tree)是将指针与long整数键值相 ...
- Linux内核中的算法和数据结构
算法和数据结构纷繁复杂,但是对于Linux Kernel开发人员来说重点了解Linux内核中使用到的算法和数据结构很有必要. 在一个国外问答平台stackexchange.com的Theoretica ...
- Linux内核中常用的数据结构和算法(转)
知乎链接:https://zhuanlan.zhihu.com/p/58087261 Linux内核代码中广泛使用了数据结构和算法,其中最常用的两个是链表和红黑树. 链表 Linux内核代码大量使用了 ...
- Linux内核分析--内核中的数据结构双向链表续【转】
在解释完内核中的链表基本知识以后,下面解释链表的重要接口操作: 1. 声明和初始化 实际上Linux只定义了链表节点,并没有专门定义链表头,那么一个链表结构是如何建立起来的呢?让我们来看看LIST_H ...
- Linux内核分析--内核中的数据结构双向链表【转】
本文转自:http://blog.csdn.net/yusiguyuan/article/details/19840065 一.首先介绍内核中链表 内核中定义的链表是双向链表,在上篇文章--libev ...
- Linux 内核中的 Device Mapper 机制
本文结合具体代码对 Linux 内核中的 device mapper 映射机制进行了介绍.Device mapper 是 Linux 2.6 内核中提供的一种从逻辑设备到物理设备的映射框架机制,在该机 ...
- Linux内核中双向链表的经典实现
概要 前面一章"介绍双向链表并给出了C/C++/Java三种实现",本章继续对双向链表进行探讨,介绍的内容是Linux内核中双向链表的经典实现和用法.其中,也会涉及到Linux内核 ...
- Linux内核中的GPIO系统之(3):pin controller driver代码分析
一.前言 对于一个嵌入式软件工程师,我们的软件模块经常和硬件打交道,pin control subsystem也不例外,被它驱动的硬件叫做pin controller(一般ARM soc的datash ...
- KSM剖析——Linux 内核中的内存去耦合
简介: 作为一个系统管理程序(hypervisor),Linux® 有几个创新,2.6.32 内核中一个有趣的变化是 KSM(Kernel Samepage Merging) 允许这个系统管理程序通 ...
随机推荐
- unity 3D循环滚动效果
https://blog.csdn.net/qinyuanpei/article/details/52765356 https://blog.csdn.net/chongzi_daima/articl ...
- 【原创】大叔经验分享(68)maven工程查看jar包依赖
1 idea 结果 2 maven命令 $ mvn dependency:tree 结果 [INFO] +- org.springframework.boot:spring-boot-starter- ...
- Const指针 、 指向const的指针 、引用、指针
1. const指针和 指向const的指针 指向const的指针: 不允许通过指针来改变其指向的const值 const double *cptr *cptr = 42; // error! 指针 ...
- mysql的decimal(10,0) not null问题
今天排查一个bug发现开发环境老是报错 order_num 字段insert的时候不能为空,但是发现测试环境没有这个问题. 后来发现测试环境有一个数据库docker安装的mysql 版本是5.7 而 ...
- JS中有两种自加法操作
JS中有两种自加法操作.它们的运算符是++,它们的函数是向1添加运算符. 我和我的区别在于操作的顺序和组合的方向. 其中:++var被称为预自动添加,变量执行自动添加操作后.它的操作是先执行自动加法操 ...
- java字符常量
在Java程序中经常会遇到类似于"Hello"这样地字符串,那么这种类型的字符串是Java中是如何存储,下面就说明字符串常量在内存中存储方式: Java程序在编译时会将程序中出现的 ...
- eclipse中svn从分支合并到主干及冲突解决
https://blog.csdn.net/shengqianfeng/article/details/79203156
- fastadmin 页面添加编辑日期时间
添加 <div class="form-group"> <label class="control-label col-xs-12 col-sm-2&q ...
- Delphi 画笔
樊伟胜
- electron api sendInputEvent 源码
electron-master\electron-master\shell\browser\api\atom_api_web_contents.cc // Copyright (c) 2014 Git ...