好湿?好紧?好多水好爽自慰,久久久噜久噜久久综合,成人做爰A片免费看黄冈,机机对机机30分钟无遮挡

主頁 > 知識庫 > linux內核編程container of()函數介紹

linux內核編程container of()函數介紹

熱門標簽:貴陽教育行業電話外呼系統 烏海智能電話機器人 做外呼系統的公司違法嗎 400電話申請方案 寧夏房產智能外呼系統要多少錢 在百度地圖標注車輛 藍點外呼系統 威海人工外呼系統供應商 撫順移動400電話申請

前言

在linux 內核編程中,會經常見到一個宏函數container_of(ptr,type,member), 但是當你通過追蹤源碼時,像我們這樣的一般人就會絕望了(這一堆都是什么呀? 函數還可以這樣定義??? 怎么還有0呢???  哎,算了,還是放棄吧。。。)。 這就是內核大佬們厲害的地方,隨便兩行代碼就讓我們懷疑人生,凡是都需要一個過程,慢慢來吧。

        其實,原理很簡單:  已知結構體type的成員member的地址ptr,求解結構體type的起始地址。

                  type的起始地址 = ptr - size      (這里需要都轉換為char *,因為它為單位字節)。

       到此,該函數已經講完,是不是很簡單??? 其實也不是,這里并沒有提到size如何計算,而令我們頭暈的正是這里。

    好吧,先上container of函數原型:

#define container_of(ptr, type, member) ({              \         
const typeof( ((type *)0)->member ) *__mptr = (ptr);    \         
(type *)( (char *)__mptr - offsetof(type,member) );})

    其次為 offserof 函數原型:

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

  怎么樣,是不是很炫?  好吧,下面開始揭開面紗:

  (一)0 指針的使用    (自己給的名字,不知有木問題)

            讓事實說話:

#include<stdio.h>
 
struct test
{
	char i ;
	int j;
	char k;
};
 
int main()
{
	struct test temp;
	printf("&temp = %p\n",&temp);   
	printf("&temp.k = %p\n",&temp.k);
	printf("&((struct test *)0)->k = %d\n",((int)&((struct test *)0)->k));
 
}

 編譯運行,可以得到如下結果:

&temp = 0xbf9815b4
&temp.k = 0xbf9815bc
&((struct test *)0)->k = 8

 什么意思看到了吧,自定義的結構體有三個變量:i,j,k。 因為有字節對齊要求,所以該結構體大小為4bytes * 3 =12 bytes.   而&((struct test *)0)->k 的作用就是求 k到結構體temp起始地址的字節數大小(就是我們的size)。在這里0被強制轉化為struct test *型, 它的作用就是作為指向該結構體起始地址的指針,就是作為指向該結構體起始地址的指針,就是作為指向該結構體起始地址的指針, 而&((struct test *)0)->k  的作用便是求k到該起始指針的字節數。。。其實是求相對地址,起始地址為0,則&k的值便是size大小(注:打印時因為需要整型,所以有個int強轉)所以我們便可以求我們需要的 size 了  。 好吧,一不小心把 offsetof() 函數的功能給講完了:::

#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

這次再看就順眼了吧(底層為什么是這樣我還是不懂。。。只知道這樣確實可以) ,  所以offsetof()的作用就是求我們夢寐以求的size, 并以size_t形式返回(size_t: 無符號整型)。

(二) 內核編程的嚴謹性  

#define container_of(ptr, type, member) ({              \         
const typeof( ((type *)0)->member ) *__mptr = (ptr);    \         
(type *)( (char *)__mptr - offsetof(type,member) );})

    這里我們只看第二行:

const typeof( ((type *)0)->member ) *__mptr = (ptr);  

  它的作用是什么呢? 其實沒什么作用(勿噴勿噴,讓我把話說完),但就形式而言 _mptr = ptr,  那為什么要要定義一個一樣的變量呢??? 其實這正是內核人員的牛逼之處:如果開發者使用時輸入的參數有問題:ptr與member類型不匹配,編譯時便會有warnning, 但是如果去掉改行,那個就沒有了,而這個警告恰恰是必須的(防止出錯有不知道錯誤在哪里)。。。這嚴謹性可以吧

typeof( ((type *)0)->member )

   它的作用是獲取member的類型僅此而已。至此基本結束

(三) 總結

       container_of(ptr, type,member)函數的實現包括兩部分:

           1.  判斷ptr 與 member 是否為同意類型

           2.  計算size大小,結構體的起始地址 = (type *)((char *)ptr - size)   (注:強轉為該結構體指針)

    現在我們知道container_of()的作用就是通過一個結構變量中一個成員的地址找到這個結構體變量的首地址。

    container_of(ptr,type,member),這里面有ptr,type,member分別代表指針、類型、成員。

到此這篇關于linux內核編程container of()函數的文章就介紹到這了,更多相關linux container of()函數 內容請搜索腳本之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持腳本之家!

標簽:那曲 朝陽 泰州 蕪湖 松原 慶陽 周口 銅川

巨人網絡通訊聲明:本文標題《linux內核編程container of()函數介紹》,本文關鍵詞  linux,內核,編程,container,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《linux內核編程container of()函數介紹》相關的同類信息!
  • 本頁收集關于linux內核編程container of()函數介紹的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 十八款app黄app入口| 曰批全过程免费视频在线观看国产| 免费一区二区三区免费视频| 舒茎馆AV导航| 欧美成人日韩| 日本一级无码爽A片免费丁给片| 久久精品电影免费动漫| 日韩a级毛片免费视频| 大学生情侣激情啪啪A片酒店| 久久香蕉成人AV大片免费看| 911精品国产一区二区| 91看片在线| 97理论片影院在线观看| 精品免费国产一区二区三区四区介绍 | 欧美xxxxzozo另类| 国产专区视频| 午夜精品久久久久99热蜜桃导演 | 男同黄网站| 史上最强内线| 爱情狂人| 亚洲护士老师的毛茸茸| yn荡小镇公交车售票员| 日本又黄又大又爽A片三年片| 全彩成人18h漫画| 国外钢针穿乳虐乳视频| 蜜w桃w传w媒w少w妇| 国内精品久久久久精品免费| 久久夜夜做天天爽国产乱辈| 小泽玛利亚一区二区三区最新章节| 你忍一下我很大| 暖暖www视频免费观看最新期| 黄色美容院在线观看| 顶级欧美熟妇XXOOHD艳星| 边啃奶头边躁狠狠躁玩爽日本| 好大好硬好爽快点我要老师| chinese快递员坚硬粗大网址| 亚洲 欧美 国产 综合美91| 丰满大乳女啪啪中文字幕| 人与野鲁XXXX毛片真播| 被cao的下不了床男男| 久久久久久久久精品|