Linux内核中链表的实现与应用【转】

简介: 转自:http://blog.chinaunix.net/uid-27037833-id-3237153.html 链表(循环双向链表)是Linux内核中最简单、最常用的一种数据结构。                1、链表的定义             struct list_head {...

转自:http://blog.chinaunix.net/uid-27037833-id-3237153.html

链表(循环双向链表)是Linux内核中最简单、最常用的一种数据结构。

       
       1、链表的定义
            struct list_head {
                struct list_head *next, *prev;
            }
           这个不含数据域的链表,可以嵌入到任何数据结构中,例如可按如下方式定义含有数据域的链表:
            struct my_list {
                void  * mydata;
                struct list_head  list;
            } ;
 
       2、链表的声明和初始化宏
            struct list_head 只定义了链表结点,并没有专门定义链表头.那么一个链表结点是如何建立起来的?
内核代码 list.h 中定义了两个宏:
            #defind  LIST_HEAD_INIT(name)    { &(name), &(name) }      //仅初始化
            #defind  LIST_HEAD(name)     struct list_head  name = LIST_HEAD_INIT(name)  //声明并初始化
           
            如果要声明并初始化链表头mylist_head,则直接调用:LIST_HEAD(mylist_head),之后,
mylist_head的next、prev指针都初始化为指向自己。这样,就有了一个带头结点的空链表。
 
             判断链表是否为空的函数:
             static inline int list_empty(const struct list_head  * head) {
                  return head->next  ==  head;
              }    //返回1表示链表为空,0表示不空
 
      3、在链表中增加一个结点
          (内核代码中,函数名前加两个下划线表示内部函数)
            static inline void   __list_add(struct list_head *new, struct list_head *prev, struct list_head *next)
            {
                     next -> prev = new ;
                     new -> next = next ;
                     new -> prev = prev ;
                     prev -> next = new ;
            }  
          
            list.h 中增加结点的两个函数为:
           (链表是循环的,可以将任何结点传递给head,调用这个内部函数以分别在链表头和尾增加结点)
            static inline void list_add(struct list_head *new, struct llist_head *head)
            {
                    __list_add(new, head, head -> next) ;
            }
            static inline void list_add_tail(struct list_head 8new, struct list_head *head)
            {
                     __list_add(new, head -> prev, head) ;
            }
            附:给head传递第一个结点,可以用来实现一个队列,传递最后一个结点,可以实现一个栈。
            static 加在函数前,表示这个函数是静态函数,其实际上是对作用域的限制,指该函数作用域仅局限
           于本文件。所以说,static 具有信息隐蔽的作用。而函数前加 inline 关键字的函数,叫内联函数,表
           示编译程序在调用这个函数时,立即将该函数展开。
            
    4、 遍历链表
           list.h 中定义了如下遍历链表的宏:
           #define   list_for_each(pos, head)    for(pos = (head)-> next ;  pos != (head) ;  pos = pos -> next)  
           这种遍历仅仅是找到一个个结点的当前位置,那如何通过pos获得起始结点的地址,从而可以引用结
点的域?list.h 中定义了 list_entry 宏:
           #define   list_entry( ptr, type, member )  \
              ( (type *) ( (char *) (ptr)  - (unsigned long) ( &( (type *)0 )  ->  member ) ) )
          分析:(unsigned long) ( &( (type *)0 )  ->  member ) 把 0 地址转化为 type 结构的指针,然后获取该
          结构中 member 域的指针,也就是获得了 member 在type 结构中的偏移量。其中  (char *) (ptr) 求
         出的是 ptr 的绝对地址,二者相减,于是得到 type 类型结构体的起始地址,即起始结点的地址。
 
   5、链表的应用
         一个用以创建、增加、删除和遍历一个双向链表的Linux内核模块

点击(此处)折叠或打开

  1. #include <linux/kernel.h>
  2. #include <linux/module.h>
  3. #include <linux/slab.h>
  4. #include <linux/list.h>
  5. MODULE_LICENCE("GPL");
  6. MODULE_AUTHOR("LUOTAIJIA");
  7. #define N 10
  8. struct numlist {
  9.     int num;
  10.     struct list_head list;
  11. };
  12. struct numlist numhead;
  13. static int __init doublelist_init(void)
  14. {
  15.     //初始化头结点
  16.     struct numlist * listnode; //每次申请链表结点时所用的指针
  17.     struct list_head * pos;
  18.     struct numlist * p;
  19.     int i;
  20.     printk("doublelist is starting...\n");
  21.     INIT_LIST_HEAD(&numhead.list);
  22.     /* 
  23.      * static inline void INIT_LIST_HEAD(struct list_head *list)
  24.      * {
  25.      * list->next = list;
  26.      * list->prev = list;
  27.      * }
  28.      */
  29.     //建立N个结点,依次加入到链表当中
  30.     for (i=0; i<N; i++) {
  31.         listnode = (struct numlist *)kmalloc(sizeof(struct numlist), GFP_KERNEL);
  32.         //void *kmalloc(size_t size, int flages) 
  33.         //分配内存,size 要分配内存大小,flags 内存类型
  34.         listnode->num = i+1;
  35.         list_add_tail(&listnode->list, &numhead.list);
  36.         printk("Node %d has added to the doublelist...\n", i+1);
  37.     }
  38.     //遍历链表
  39.     i = 1;
  40.     list_for_each(pos, &numhead.list) {
  41.         p = list_entry(pos, struct numlist, list);
  42.         printk("Node %d's data: %d\n", i, p->num);
  43.         i++;
  44.     }
  45.     return 0;
  46. }
  47. static void __exit doublelist_exit(void)
  48. {
  49.     struct list_head *pos, *n;
  50.     struct numlist *p;
  51.     int i;
  52.     //依次删除N个结点
  53.     i = 1;
  54.     list_for_each_safe(pos, n, &numhead.list) {
  55.         //为了安全删除结点而进行的遍历
  56.         list_del(pos); //从链表中删除当前结点
  57.         p = list_entry(pos, struct numlist, llist); 
  58.         //得到当前数据结点的首地址,即指针
  59.         kfree(p); //释放该数据结点所占空间
  60.         printk("Node %d has removed from the doublelist...\n", i++);
  61.     }
  62.     printk("doublelist is exiting...\n");
  63. }
  64. module_init(doublelist_init);
  65. module_exit(doublelist_exit);
 
参考资料:Linux操作系统原理与应用(第2版)    陈莉君、康华 编著
【作者】 张昺华
【新浪微博】 张昺华--sky
【twitter】 @sky2030_
【facebook】 张昺华 zhangbinghua
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利.
目录
相关文章
|
3天前
|
云安全 人工智能 安全
AI被攻击怎么办?
阿里云提供 AI 全栈安全能力,其中对网络攻击的主动识别、智能阻断与快速响应构成其核心防线,依托原生安全防护为客户筑牢免疫屏障。
|
13天前
|
域名解析 人工智能
【实操攻略】手把手教学,免费领取.CN域名
即日起至2025年12月31日,购买万小智AI建站或云·企业官网,每单可免费领1个.CN域名首年!跟我了解领取攻略吧~
|
7天前
|
安全 Java Android开发
深度解析 Android 崩溃捕获原理及从崩溃到归因的闭环实践
崩溃堆栈全是 a.b.c?Native 错误查不到行号?本文详解 Android 崩溃采集全链路原理,教你如何把“天书”变“说明书”。RUM SDK 已支持一键接入。
509 203
|
5天前
|
人工智能 移动开发 自然语言处理
2025最新HTML静态网页制作工具推荐:10款免费在线生成器小白也能5分钟上手
晓猛团队精选2025年10款真正免费、无需编程的在线HTML建站工具,涵盖AI生成、拖拽编辑、设计稿转代码等多种类型,均支持浏览器直接使用、快速出图与文件导出,特别适合零基础用户快速搭建个人网站、落地页或企业官网。
685 157
|
11天前
|
人工智能 自然语言处理 安全
国内主流Agent工具功能全维度对比:从技术内核到场景落地,一篇读懂所有选择
2024年全球AI Agent市场规模达52.9亿美元,预计2030年将增长至471亿美元,亚太地区增速领先。国内Agent工具呈现“百花齐放”格局,涵盖政务、金融、电商等多场景。本文深入解析实在智能实在Agent等主流产品,在技术架构、任务规划、多模态交互、工具集成等方面进行全维度对比,结合市场反馈与行业趋势,为企业及个人用户提供科学选型指南,助力高效落地AI智能体应用。
|
5天前
|
数据采集 消息中间件 人工智能
跨系统数据搬运的全方位解析,包括定义、痛点、技术、方法及智能体解决方案
跨系统数据搬运打通企业数据孤岛,实现CRM、ERP等系统高效互通。伴随数字化转型,全球市场规模超150亿美元,中国年增速达30%。本文详解其定义、痛点、技术原理、主流方法及智能体新范式,结合实在Agent等案例,揭示从数据割裂到智能流通的实践路径,助力企业降本增效,释放数据价值。
|
存储 人工智能 监控
从代码生成到自主决策:打造一个Coding驱动的“自我编程”Agent
本文介绍了一种基于LLM的“自我编程”Agent系统,通过代码驱动实现复杂逻辑。该Agent以Python为执行引擎,结合Py4j实现Java与Python交互,支持多工具调用、记忆分层与上下文工程,具备感知、认知、表达、自我评估等能力模块,目标是打造可进化的“1.5线”智能助手。
673 46