在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

【Linux內核】從小小的宏定義窺探Linux內核的精妙設計

嵌入式物聯網開發 ? 來源:嵌入式物聯網開發 ? 作者:嵌入式物聯網開發 ? 2022-08-31 13:30 ? 次閱讀
 Linux操作系統,可以說它就是程序猿的代碼天堂;這不僅僅因為它是開源的,更多的是因為它的誕生,是由世界上無數的代碼天才共同締造而來;跑在它上面的Linux內核,經受了世界上各式各樣的服務器壓力測試,始終保持著高效、穩定、安全的特性,一如既往地服務全人類。甚至可以說Linux操作系統造福了人類,很難想象,當Linux操作系統消失了,這個世界會變得怎么樣?

? 作為Linux操作系統的忠實粉絲,筆者自大學時期就開始研究和使用Linux操作系統,出來工作了好幾年,幾乎每天都要跟Linux系統打交道,甚至毫不夸張的是,白天不在Linux系統命令行下敲幾行命令,晚上都會失眠。

? 學習和使用了Linux系統這么些年,一直想找個機會,對Linux的知識做一番梳理,無奈之前礙于各種時間因素和自我的惰性,遲遲未有實質性的進展。最近才開始狠狠地下定決心,必須邁出扎實的一步,爭取做出更多的分享,充實自我的同時,也給同行帶來更多的視野和思路,何樂而不為呢?


? 本文打算從一個很小的代碼設計,試圖從中窺探一下Linux內核代碼的精妙設計。它的名字就叫 max宏定義,請跟隨筆者的思路一步步解開它神秘的面紗。

? 先來一個它的全貌:

#define max(a, b) ({\
typeof(a) _max1 = (a);\
typeof(b) _max2 = (b);\
(void)(&_max1 == &_max2);\
_max1 > _max2 ? _max1 : _max2; })

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-AQSFb6aB-1661923667090)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 我們先不一下子就把這段代碼剖析徹底,換個思維,假設我們是Linux內核的設計者,要解決比較2個數的大小,代碼應該怎么樣入手。我想很多C語言工作者,甚至是初學C語言的碼農也可以寫出這樣的如下代碼:

#define max(a, b) a > b ? a : b

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-jvjzDDsR-1661923667093)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 初看這個宏定義,似乎沒有問題;細細一看,用個測試案例一測試就發現端倪了:

/* 假設有如下的調用代碼 */
{    
    printf("result = %d\n", max(9!=9, 0==0));

    /* 宏定義展開后是  9!=9 > 0==0 ? 9!=9 : 0==0*/
    /* 輸出結果是 0*/
}

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-59Bp6ggT-1661923667094)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 很明顯正確答案應該是輸出1,細心者就很快發現,給a和b加上括號試試看:

#define max(a, b) (a) > (b) ? (a) : (b)

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-k7e0TQdq-1661923667097)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 調用代碼測試下:

/* 假設有如下的調用代碼 */
{
    printf("result = %d\n", max(9!=9, 0==0));

    /* 宏定義展開后是  (9!=9) > (0==0) ?(9!=9):(0==0)*/
    /* 輸出正確結果 1*/

    printf("result = %d\n", 9 + max(9!=9, 0==0));

    /* 宏定義展開后是  9 + (9!=9) > (0==0) ?(9!=9 :(0==0)*/
    /* 輸出結果是0, 正確的期望值輸出,應該是10 (=9+1) */
}

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-m705RxoN-1661923667114)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 于是又有了下面的改進:

#define max(a, b) ((a) > (b) ? (a) : (b))

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-xriR0pv8-1661923667116)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 這個版本,也是我們日常寫代碼最經常看到的版本,我們使用測試代碼測試下看看:

/* 假設有如下的調用代碼 */
{
    printf("result = %d\n", 9 + max(9!=9, 0==0));

    /* 宏定義展開后是  9 + ((9!=9) > (0==0) ?(9!=9):(0==0))*/
    /* 輸出正確的期望值10 (=9+1) */

    int a = 8;
    int b = 9;

    printf("result = %d\n", max(a++, b++));

    /* 宏定義展開后是  ((a++) > (b++) ?(a++):(b++))*/
    /* 輸出結果是10;而正確的期望值輸出,應該是9 */
}

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-2JAec1EG-1661923667122)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 很遺憾,經測試,這個版本依然有問題,這是因為宏定義中的++操作干擾了比較結果的輸出,我們需要再次改進這個宏定義。應該怎么樣改進呢?既然是++操作干擾了輸出,那么我們使用2個中間變量來中轉下不就ok了嗎?于是有了下面的版本:

#define max(a, b) ({\
int _a = (a);\
int _b = (b);\
_a > _b ? _a : _b;\
})

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-t7VD8Szr-1661923667123)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 這樣的寫法,已經有點接近Linux內核定義的模樣了。再次使用上面的測試代碼執行測試:

/* 假設有如下的調用代碼 */
{    
    int a = 8;
    int b = 9;

    printf("result = %d\n", max(a++, b++));

    /* 宏定義展開后是  ({int _a=a++; int _b=b++; _a > _b ? _a : _b;})*/
    /* 輸出正確的期望值9 */
}

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-dpinR0OT-1661923667125)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 雖然上面版本的定義解決了++操作符引起的輸出結果錯誤的問題,但是由于宏定義內部使用了int型的_a和_b作為中間變量,這就是限制了max宏定義只能用于2個int型的數據做比較,這將大大限制了它的使用范圍。于是,很容易想到一個解決辦法,將int這個數據類型使用type變量傳進去,于是有了下面的版本:

#define max(type, a, b) ({\
type _a = (a);\
type _b = (b);\
_a > _b ? _a : _b;\
})

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-WVFbRavF-1661923667126)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 這樣的確是解決了如上數據類型問題的困惑,但是這樣我們的宏定義是以多一個參數輸入為犧牲代價的。那么,我們有沒有什么辦法,可以不將type輸入,而直接從輸入的a和b中獲取它們的數據類型呢?答案是肯定有的!

? GNU C作為C語言的擴展版本,增加了若干非常有用的擴展語法,其中typeof關鍵字就是其中的一個。比如定義一個變量int a; 則typeof(a)就可以取得a變量的類型,即int;比如直接使用typeof(unsigned char *),得到的輸出就是數據類型unsigned char *,非常的實用。于是我們將typeof應用到max宏中,于是就有下面的優良版本:

#define max(a, b) ({\
typeof(a) _a = (a);\
typeof(a) _b = (b);\
_a > _b ? _a : _b;\
})

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-KTnbdo0C-1661923667127)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 這樣的寫法,雖然避免了我們傳遞a和b變量的數據類型進去,但是,如下的測試代碼,結果會怎么樣呢?

/* 假設有如下的調用代碼 */
{    
    int a = 8;
    float b = 9.0;

    printf("result = %d\n", max(a, b));
    /* 這樣能比較嗎?*/

    int a = 8;
    float b = 9.0;
    float *p = &b;

    printf("result = %d\n", max(a, p));
    /* 這樣又能比較嗎?*/
}

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-j0v9A1VG-1661923667128)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 很明顯,當a是int型,而b是float型,內核執行比較是可以的;但是如果拿一個int型的變量跟一個float *變量做比較,或者兩個奇奇怪怪的struct類型變量做計較,這樣肯定是不行的。所以,我們在設計max宏定義的時候,需要將這種可能出現的問題盡可能地在編譯階段就暴露出來,于是有了Linux內核max宏定義的最佳版本:

#define max(a, b) ({\
typeof(a) _a = (a);\
typeof(a) _b = (b);\
(void) &_a == &_b;\
_a > _b ? _a : _b;\
})

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-YVJm8AAy-1661923667129)(data:image/gif;base64,R0lGODlhAQABAPABAP///wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==)]

? 我們注意,宏定義的第4行,(void) &_a == &_b; 意思是對_a和_b的地址做比較,實際上從運行結果上,這個肯定是不等的,但是我們關心的并不是兩者比較的結果,而是兩者能不能用==比較的問題。當_a和_b的數據類型一致時,代碼編譯不會有任何警告;反之,當兩者的數據類型不一致時,比如之前的a是int型,而b是float型,那么這條語句就會報出編譯警告,如果在嚴格的編譯選項下,這個警告還可以轉換為錯誤,要求代碼調用者去確認結果,是否對兩個不同類型的數據執行max比較的動作,從而將隱患消除,提升代碼質量。


? 通過跟隨筆者的思路,我們可以細細地體會到,內核設計者在設計這個max宏時,相信也是走了不少的彎路,從一開始最簡版本,接著遇到各式各樣的問題,然后一步步解決,一步步完善設計,最終才有最優秀的max宏呈現在我們面前。如此之類的代碼設計,在Linux內核設計代碼中比比皆是,今后筆者也會集中整理此類的優秀設計,致力于將更多的優秀內核代碼分享給讀者,敬請關注。文中提及的觀點,均為筆者愚見,如有紕漏之處,還望誠心指正,謝謝。

審核編輯:湯梓紅
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 內核
    +關注

    關注

    3

    文章

    1382

    瀏覽量

    40423
  • Linux
    +關注

    關注

    87

    文章

    11345

    瀏覽量

    210395
  • 操作系統
    +關注

    關注

    37

    文章

    6895

    瀏覽量

    123745
收藏 人收藏

    評論

    相關推薦

    Linux內核中C語言的使用技巧

    Linux內核可謂是集C語言大成者,從中我們可以學到非常多的技巧,本文來學習一下技巧,文章有點長,但耐心看完后C語言level直接飆升。
    發表于 07-21 14:56 ?503次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>內核</b>中C語言<b class='flag-5'>宏</b>的使用技巧

    Linux內核中文版教程

    Linux內核中文版教程
    發表于 03-28 09:45 ?0次下載

    Linux內核教程

    本章學習目標掌握LINUX內核版本的含義理解并掌握進程的概念掌握管道的概念及實現了解內核的數據結構了解LINUX內核的算法掌握
    發表于 04-10 16:59 ?0次下載

    linux內核的完全注釋

    linux內核的完全注釋
    發表于 10-29 10:02 ?18次下載
    <b class='flag-5'>linux</b><b class='flag-5'>內核</b>的完全注釋

    Linux_內核注釋

    Linux_內核注釋
    發表于 10-30 09:45 ?9次下載
    <b class='flag-5'>Linux</b>_<b class='flag-5'>內核</b>注釋

    Linux內核編譯詳談

    Linux內核編譯詳談
    發表于 10-30 09:51 ?7次下載
    <b class='flag-5'>Linux</b><b class='flag-5'>內核</b>編譯詳談

    Linux內核配置系統詳解

    隨著 Linux 操作系統的廣泛應用,特別是 Linux 在嵌入式領域的發展,越來越多的人開始投身到 Linux 內核級的開發中。面對日益龐大的 L
    發表于 11-01 15:45 ?4次下載

    linux內核是什么_linux內核學習路線

    Linux內核是一個操作系統(OS)內核,本質上定義為類Unix。它用于不同的操作系統,主要是以不同的Linux發行版的形式。
    發表于 09-16 15:49 ?2675次閱讀

    linux內核參數設置_linux內核的功能有哪些

    本文主要闡述了linux內核參數設置及linux內核的功能。
    發表于 09-17 14:40 ?1402次閱讀
    <b class='flag-5'>linux</b><b class='flag-5'>內核</b>參數設置_<b class='flag-5'>linux</b><b class='flag-5'>內核</b>的功能有哪些

    最硬核的Linux內核文章

    來源 :頭條號@Linux學習教程,冰凌塊兒 01 前言 本文主要講解什么是Linux內核,以及通過多張圖片展示Linux內核的作用與功能,
    的頭像 發表于 10-19 17:46 ?2161次閱讀
    最硬核的<b class='flag-5'>Linux</b><b class='flag-5'>內核</b>文章

    快速理解什么是Linux內核以及Linux內核的內容

    01 前言 本文主要講解什么是Linux內核,以及通過多張圖片展示Linux內核的作用與功能,以便于讀者能快速理解什么是Linux
    的頭像 發表于 10-21 12:02 ?4337次閱讀
    快速理解什么是<b class='flag-5'>Linux</b><b class='flag-5'>內核</b>以及<b class='flag-5'>Linux</b><b class='flag-5'>內核</b>的內容

    Linux 5.10.5內核正式發布

    1月6日,Linux基金會宣布,Linux 5.10.5內核正式發布,所有5.10內核系列的用戶都必須升級。
    的頭像 發表于 01-07 14:36 ?2642次閱讀

    嵌入式LINUX系統內核內核模塊調試

    嵌入式LINUX系統內核內核模塊調試(嵌入式開發和硬件開發)-嵌入式LINUX系統內核內核
    發表于 07-30 13:55 ?10次下載
    嵌入式<b class='flag-5'>LINUX</b>系統<b class='flag-5'>內核</b>和<b class='flag-5'>內核</b>模塊調試

    Linux內核文件Cache機制

    Linux內核文件Cache機制(開關電源技術與設計 第二版)-Linux內核文件Cache機制? ? ? ? ? ? ? ??
    發表于 08-31 16:34 ?4次下載
    <b class='flag-5'>Linux</b><b class='flag-5'>內核</b>文件Cache機制

    基于Android的Linux內核的電源管理:概述

    1.電源管理的狀態Android的Linux內核為系統提供了4種電源狀態,內核的源代碼為其中的3種定義了名字和對應的
    發表于 01-07 11:14 ?6次下載
    基于Android的<b class='flag-5'>Linux</b><b class='flag-5'>內核</b>的電源管理:概述
    主站蜘蛛池模板: 999久久久免费精品国产牛牛 | 伦理一区二区三区 | 国产午夜精品久久久久免费视 | 国产免费人成在线看视频 | 日本一卡二卡≡卡四卡精品 | 色偷偷免费视频 | 亚洲精品在线不卡 | 19xxxxxxxxx日本69| 男人天堂网在线视频 | 久久夜色精品国产亚洲噜噜 | 美女视频大全美女视频黄 | 美女屁屁免费视频网站 | 一区二区影视 | 天天干夜夜爽 | 久久久久久久久综合 | 天天爱天天操 | 2021久久精品国产99国产 | 午夜视频黄 | 激情综合色综合啪啪开心 | 日本黄色免费看 | 精品一区二区三区在线视频 | 一区二区三区四区精品 | 天天爱天天做天天爽天天躁 | 手机在线视频你懂的 | 日本色www| 美国一级大黄香蕉片 | 欧美xxxx性特级高清 | 国产黄色网页 | 国产高清亚洲 | 四虎国产精品免费入口 | 欧美日本视频一区 | www.午夜剧场| 日韩精品无码一区二区三区 | 中文在线资源链接天堂 | 国产精品视频永久免费播放 | 在线观看日本免费不卡 | 中文天堂网在线www 中文天堂资源在线www | 免费啪视频观在线视频在线 | 九色视频在线看 | 77ee成人 | 国产成人午夜精品影院游乐网 |