欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Linux 內(nèi)核通用鏈表學習小結(jié)

 更新時間:2017年11月07日 11:57:18   作者:簡單方式  
本篇文章主要介紹了Linux 內(nèi)核通用鏈表學習小結(jié),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

描述

在linux內(nèi)核中封裝了一個通用的雙向鏈表庫,這個通用的鏈表庫有很好的擴展性和封裝性,它給我們提供了一個固定的指針域結(jié)構(gòu)體,我們在使用的時候,只需要在我們定義的數(shù)據(jù)域結(jié)構(gòu)體中包含這個指針域結(jié)構(gòu)體就可以了,具體的實現(xiàn)、鏈接并不需要我們關心,只要調(diào)用提供給我們的相關接口就可以完成了。

傳統(tǒng)的鏈表結(jié)構(gòu)

struct node{
  int key;
  int val;
  node* prev;
  node* next;
 }

linux 內(nèi)核通用鏈表庫結(jié)構(gòu)

提供給我們的指針域結(jié)構(gòu)體:

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

我們只需要包含它就可以:

struct node{
  int val;
  int key;
  struct list_head* head;
}

可以看到通過這個 list_head 結(jié)構(gòu)就把我們的數(shù)據(jù)層跟驅(qū)動層分開了,而內(nèi)核提供的各種操作方法接口也只關心 list_head 這個結(jié)構(gòu),也就是具體鏈接的時候也只鏈接這個list_head 結(jié)構(gòu),并不關心你數(shù)據(jù)層定義了什么類型.

一些接口宏定義

//初始化頭指針
#define LIST_HEAD_INIT(name) { &(name), &(name) }

#define LIST_HEAD(name) \
  struct list_head name = LIST_HEAD_INIT(name)

//遍歷鏈表
#define __list_for_each(pos, head) \
  for (pos = (head)->next; pos != (head); pos = pos->next)

//獲取節(jié)點首地址(不是list_head地址,是數(shù)據(jù)層節(jié)點首地址)
#define list_entry(ptr, type, member) \
  container_of(ptr, type, member)

//container_of在Linux內(nèi)核中是一個常用的宏,用于從包含在某個
//結(jié)構(gòu)中的指針獲得結(jié)構(gòu)本身的指針,通俗地講就是通過結(jié)構(gòu)體變
//量中某個成員的首地址進而獲得整個結(jié)構(gòu)體變量的首地址
#define container_of(ptr, type, member) ({     \
    const typeof( ((type *)0)->member ) *__mptr = (ptr);  \
    (type *)( (char *)__mptr - offsetof(type,member) );})

#define offsetof(s,m) (size_t)&(((s *)0)->m)

使用方式

typedef struct node{
  int val;
  int key;
  struct list_head* list;
}node;

//初始化頭指針
LIST_HEAD(head);

//創(chuàng)建節(jié)點
node* a = malloc(sizeof(node));
node* b = malloc(sizeof(node));

//插入鏈表 方式一
list_add(&a->list,&head);
list_add(&b->list,&head);

//插入鏈表 方式二
list_add_tail(&a->list,&head);
list_add_tail(&b->list,&head);

//遍歷鏈表  
struct list_head* p;
struct node* n;
__list_for_each(p,head){
  //返回list_head地址,然后再通過list_head地址反推
  //節(jié)點結(jié)構(gòu)體首地址.
  n = list_entry(pos,struct node,list);
}

list_add 接口,先入后出原則,有點類似于棧

list_add-先入后出模式

list_add_tail 接口,先入先出原則,有點類似于fifo

list_add-先入先出模式

我們的鏈表節(jié)點,實際在內(nèi)存中的展示形態(tài)

節(jié)點描述

可以看到最終的形態(tài)是,通過指向每個結(jié)構(gòu)體里面的 list_head 類型指針,然后把它們串聯(lián)起來的

list_entry 接口,通過結(jié)構(gòu)體變量某個成員的地址,反推結(jié)構(gòu)體首地址,就像 __list_for_each 接口只返回 list_head 地址,所以我們要通過這個成員地址在去獲取它本身的結(jié)構(gòu)體首地址,底層實現(xiàn)方法 container_of 宏

反推結(jié)構(gòu)體首地址

舉個例子

這個例子包括簡單的增、刪、遍歷

#include <linux/kernel.h> 
#include <linux/module.h> 
#include <linux/init.h> 
#include <linux/slab.h> 
#include <linux/list.h> 
 
MODULE_LICENSE("GPL"); 
MODULE_AUTHOR("David Xie"); 
MODULE_DESCRIPTION("List Module"); 
MODULE_ALIAS("List module"); 
 
struct student //代表一個實際節(jié)點的結(jié)構(gòu) 
{ 
  char name[100]; 
  int num; 
  struct list_head list;  //內(nèi)核鏈表里的節(jié)點結(jié)構(gòu) 
}; 
 
struct student *pstudent;    
struct student *tmp_student; 
struct list_head student_list;  
struct list_head *pos; 
 
int mylist_init(void) 
{ 
  int i = 0; 
   
  //初始化一個鏈表,其實就是把student_list的prev和next指向自身 
  INIT_LIST_HEAD(&student_list);  
   
  pstudent = kmalloc(sizeof(struct student)*5,GFP_KERNEL);//向內(nèi)核申請5個student結(jié)構(gòu)空間 
  memset(pstudent,0,sizeof(struct student)*5); //清空,這兩個函數(shù)可以由kzalloc單獨做到 
   
  for(i=0;i<5;i++) 
  { //為結(jié)構(gòu)體屬性賦值 
    sprintf(pstudent[i].name,"Student%d",i+1); 
    pstudent[i].num = i+1;  
    //加入鏈表節(jié)點,list_add的話是在表頭插入,list_add_tail是在表尾插入 
    list_add( &(pstudent[i].list), &student_list);//參數(shù)1是要插入的節(jié)點地址,參數(shù)2是鏈表頭地址 
  }  
   
  list_for_each(pos,&student_list) //list_for_each用來遍歷鏈表,這是個宏定義 
                   //pos在上面有定義 
  { 
    //list_entry用來提取出內(nèi)核鏈表節(jié)點對應的實際結(jié)構(gòu)節(jié)點,即根據(jù)struct list_head來提取struct student 
    //第三個參數(shù)list就是student結(jié)構(gòu)定義里的屬性list 
    //list_entry的原理有點復雜,也是linux內(nèi)核的一個經(jīng)典實現(xiàn),這個在上面那篇鏈接文章里也有講解 
    tmp_student = list_entry(pos,struct student,list); 
    //打印一些信息,以備驗證結(jié)果 
    printk("<0>student %d name: %s/n",tmp_student->num,tmp_student->name); 
  } 
   
  return 0; 
} 
 
 
void mylist_exit(void) 
{   
  int i ; 
  /* 實驗:將for換成list_for_each來遍歷刪除結(jié)點,觀察要發(fā)生的現(xiàn)象,并考慮解決辦法 */ 
  for(i=0;i<5;i++) 
  { 
    //額,刪除節(jié)點,只要傳個內(nèi)核鏈表節(jié)點就行了 
    list_del(&(pstudent[i].list));    
  } 
  //釋放空間 
  kfree(pstudent); 
} 
 
module_init(mylist_init); 
module_exit(mylist_exit); 

結(jié)束

linux 內(nèi)核提供的這個通用鏈表庫里面還有很多其他的接口,這里沒有詳細的一一舉例,有興趣的可以自己去看看,在源碼包 include/linux/list.h 文件里面,不過通過閱讀一些源代碼確實對我們也有很大的提高,可以看看高手是如何去設計并實現(xiàn),還可以學到一些技巧以及對代碼細節(jié)的掌握~~.

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • Linux 塊設備驅(qū)動代碼編寫

    Linux 塊設備驅(qū)動代碼編寫

    這篇文章主要介紹了Linux 塊設備驅(qū)動代碼編寫,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-04-04
  • 解決linux下vim中文亂碼的方法

    解決linux下vim中文亂碼的方法

    在vim中編輯一個中文文本時,有時候看起來有亂碼,以前都是修修補補的弄沒有亂碼了就不管了,這個問題一直都很困擾我。突然想到這個問題想把它給解決掉,在網(wǎng)上有很多這方面的資料,但是說得不是很到位,經(jīng)過了一天的折騰并做了一些小測試終于搞定了。下面來一起看看吧。
    2016-12-12
  • Linux新建虛擬機Ubuntu的圖文教程

    Linux新建虛擬機Ubuntu的圖文教程

    這篇文章主要詳細介紹了Linux新建虛擬機Ubuntu的教程,文章通過圖文結(jié)合的方式給大家介紹的非常詳細,對大家的學習或工作有一定的幫助,需要的朋友可以參考下
    2024-09-09
  • Linux如何實現(xiàn)斷點續(xù)傳文件功能

    Linux如何實現(xiàn)斷點續(xù)傳文件功能

    最近在工作中遇到一個需求,要實現(xiàn)Linux下的文件傳輸,支持斷點續(xù)傳,所以這篇文章主要給大家介紹了關于Linux如何實現(xiàn)斷點續(xù)傳文件功能的相關資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-12-12
  • CentOS下使用LibreOffice實現(xiàn)文檔格式的轉(zhuǎn)換方式

    CentOS下使用LibreOffice實現(xiàn)文檔格式的轉(zhuǎn)換方式

    項目需求,對上傳的文檔進行一些預處理,如果用戶上傳了doc格式的文檔,需要將其處理為docx或者pdf格式,以便后續(xù)的流程對文檔內(nèi)容進行提取。接下來通過本文給大家分享CentOS下使用LibreOffice實現(xiàn)文檔格式的轉(zhuǎn)換,感興趣的朋友一起看看吧
    2019-07-07
  • SpringBoot整合Activiti7的實現(xiàn)代碼

    SpringBoot整合Activiti7的實現(xiàn)代碼

    這篇文章主要介紹了SpringBoot整合Activiti7的實現(xiàn)代碼,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-11-11
  • Win10 安裝Linux ubuntu-18.04雙系統(tǒng)(安裝指南)

    Win10 安裝Linux ubuntu-18.04雙系統(tǒng)(安裝指南)

    這篇文章主要介紹了Win10+Linux ubuntu-18.04雙系統(tǒng)安裝教程,本文分步驟給大家記錄下來,需要的朋友可以參考下
    2019-10-10
  • 把windows下的字體安裝到Linux系統(tǒng)下的方法介紹

    把windows下的字體安裝到Linux系統(tǒng)下的方法介紹

    Linux(Fedora/Ubuntu/CentOS)的字體實在不盡如人意,而且在網(wǎng)頁及文檔顯示時很多字無法顯示出來,特別多的空白和亂碼,其實,我們可以把windows下的字體和自己心儀的字體添加到Linux中,本文將介紹如何在Linux下添加字體
    2018-03-03
  • 詳解linux 關機命令總結(jié)

    詳解linux 關機命令總結(jié)

    本篇文章主要介紹了linux 關機命令總結(jié),linux下常用的關機命令有:shutdown、halt、poweroff、init,有興趣的可以了解一下。
    2016-12-12
  • Centos8搭建基于kdc加密的nfs

    Centos8搭建基于kdc加密的nfs

    大家好,本篇文章主要講的是Centos8搭建基于kdc加密的nfs,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽
    2021-12-12

最新評論