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

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

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

3天內不再提示

關于llist.h文件中的鏈表宏講解

xCb1_yikoulinux ? 來源:一口Linux ? 作者:一口Linux ? 2022-07-01 11:58 ? 次閱讀

水積春塘晚,陰交夏木繁 ——唐·白居易

鏈表宏linux內核鴻蒙內核rtos和一些開源代碼中用的非常多。鏈表宏是雙向鏈表的經典實現方式,總代碼不超過50行,相當精煉。在一些開源框架中,它的數據結構,就是以鏈表宏為基礎進行搭建(如shttpd,一個開源的輕量級、嵌入式服務器框架)。本篇文章將對llist.h文件中的鏈表宏進行逐個講解。

1 源碼(llist.h)

llist.h文件的全部源碼如下:

#ifndefLLIST_HEADER_INCLUDED
#defineLLIST_HEADER_INCLUDED

/*
*Linkedlistmacros.
*/
structllhead{
structllhead*prev;
structllhead*next;
};

#defineLL_INIT(N)((N)->next=(N)->prev=(N))

#defineLL_HEAD(H)structllheadH={&H,&H}

#defineLL_ENTRY(P,T,N)((T*)((char*)(P)-offsetof(T,N)))

#defineLL_ADD(H,N)
do{
((H)->next)->prev=(N);
(N)->next=((H)->next);
(N)->prev=(H);
(H)->next=(N);
}while(0)

#defineLL_TAIL(H,N)
do{
((H)->prev)->next=(N);
(N)->prev=((H)->prev);
(N)->next=(H);
(H)->prev=(N);
}while(0)

#defineLL_DEL(N)
do{
((N)->next)->prev=((N)->prev);
((N)->prev)->next=((N)->next);
LL_INIT(N);
}while(0)

#defineLL_EMPTY(N)((N)->next==(N))

#defineLL_FOREACH(H,N)for(N=(H)->next;N!=(H);N=(N)->next)

#defineLL_FOREACH_SAFE(H,N,T)
for(N=(H)->next,T=(N)->next;N!=(H);
N=(T),T=(N)->next)

#endif/*LLIST_HEADER_INCLUDED*/

2 注解

在llist.h中,所用到的鏈表是雙向鏈表,其節點結構定義如下。在此節點結構中,其只包含了兩個指針域,一個指向直接前驅,一個指向直接后繼,沒有定義數據域。

structllhead{
structllhead*prev;
structllhead*next;
};

2.1 LL_INIT(N)

宏LL_INIT的定義如下,其作用是將所傳入指針N的兩個指針域(N)->next和(N)->prev都指向N。目的是完成單個節點的初始化工作,如下圖示意了該過程。a65a4bbc-f8f1-11ec-ba43-dac502259ad0.png

#defineLL_INIT(N)((N)->next=(N)->prev=(N))

2.2 LL_HEAD(H)

宏LL_HEAD的定義如下,直接將宏LL_HEAD展開,其意圖很明顯是定義一個新鏈表H(H表示為傳入宏的參數名),并且將H的兩個指針域,都初始化為H地址本身,如下圖示意了該過程。a660f502-f8f1-11ec-ba43-dac502259ad0.png

#defineLL_HEAD(H)structllheadH={&H,&H}

2.3 LL_ENTRY(P,T,N)

宏LL_ENTRY的定義如下,其依賴于宏offsetof。下面先對宏offsetof進行詳細描述,其功能描述為:

C語言的offsetof()宏,是定義在stddef.h。用于求出一個struct或union數據類型的給定成員的size_t類型的字節偏移值(相對于struct或union數據類型的開頭)。offsetof()宏有兩個參數,分別是結構名與結構內的成員名。——維基百科

#defineLL_ENTRY(P,T,N)((T*)((char*)(P)-offsetof(T,N)))

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

為了更好的理解宏offsetof,下面按照宏的定義來進行拆解說明。

((TYPE *)0):取整數零并將其強轉換為指向TYPE的指針。

((TYPE *)0)->MEMBER):引用指向結構成員MEMBER。

&((TYPE *)0)->MEMBER):取出MEMBER的地址。

((size_t) &((TYPE *)0)->MEMBER):將結果轉換為適當的數據類型。

由于該結構體是以0地址開頭,所以最后該宏返回的結果就是該成員相對于結構體開頭的偏移量。有了對宏offsetof的理解,再來看宏LL_ENTRY就比較好理解了。宏LL_ENTRY的功能是,根據結構體變量(T)中的域成員變量(N)的指針(P)來獲取指向整個結構體變量的指針,下面來做拆解說明:

offsetof(T, N):計算成員N相對于其結構體T開頭的偏移量。

((char *)(P):將指針P強轉為字符指針類型,保證其做+/-運算時是以字節為單位。

(char *)(P) - offsetof(T, N)):P為成員N的指針,減去偏移量,指針到了結構體開頭位置。

((T *)((char *)(P)- offsetof(T, N))):將指針強轉,得到了整個結構體指針。

宏LL_ENTRY的作用和linux中的宏container_of作用基本一樣,該宏定義如下:

#definecontainer_of(ptr,type,member)({
consttypeof(((type*)0)->member)*__mptr=(ptr);
(type*)((char*)__mptr-offsetof(type,member));})

2.4 LL_ADD(H, N)

宏LL_ADD的定義如下,其作用是向雙向鏈表H的頭部添加節點N。根據LL_ADD定義的語句順序,對照著圖片分析,會更加清晰。如下圖,上面這張圖片展示了添加節點N之前的結構,下圖展示了添加節點N之后的結構。a66c3764-f8f1-11ec-ba43-dac502259ad0.pnga67d99fa-f8f1-11ec-ba43-dac502259ad0.png

#defineLL_ADD(H,N)
do{
((H)->next)->prev=(N);
(N)->next=((H)->next);
(N)->prev=(H);
(H)->next=(N);
}while(0)

2.5 LL_TAIL(H, N)

宏LL_TAIL的定義如下,其作用是將節點N添加到雙向鏈表H的尾部。宏LL_TAIL的定義如下,其作用是向雙向鏈表H的頭部添加節點N。根據LL_TAIL定義的語句順序,對照著圖片分析,會更加清晰。如下圖,上面這張圖片展示了添加節點N之前的結構,下圖展示了添加節點N之后的結構,可以和LL_ADD的結果進行對照。

#defineLL_TAIL(H,N)
do{
((H)->prev)->next=(N);
(N)->prev=((H)->prev);
(N)->next=(H);
(H)->prev=(N);
}while(0)

2.6 LL_DEL(N)

宏LL_DEL的定義如下,其作用是將節點N從雙向鏈表中刪除,并且節點N回到初始狀態(其指針僅指向自身,不再指向其它地方)。

#defineLL_DEL(N)
do{
((N)->next)->prev=((N)->prev);
((N)->prev)->next=((N)->next);
LL_INIT(N);
}while(0)

2.7 LL_EMPTY(N)

宏LL_EMPTY的定義如下,其作用是判斷鏈表N是否為空鏈表,返回布爾值false/true。如果節點的直接后繼next指向其自身,就認為其為空節點。

#defineLL_EMPTY(N)((N)->next==(N))

2.8 LL_FOREACH(H,N)

宏LL_FOREACH的定義如下,其作用是在雙向鏈表H中,循環遍歷出節點。

#defineLL_FOREACH(H,N)for(N=(H)->next;N!=(H);N=(N)->next)

2.9 LL_FOREACH_SAFE(H,N,T)

宏LL_FOREACH_SAFE的定義如下,其作用是在雙向鏈表H中,循環遍歷出節點N,因為其有提前存儲N的下一個節點T。即使N節點被清理掉,也不影響其下一個節點的遍歷,所以該宏一般用來做循環清除雙向鏈表中節點的操作,而宏LL_FOREACH僅用來遍歷雙向鏈表。

#defineLL_FOREACH_SAFE(H,N,T)
for(N=(H)->next,T=(N)->next;N!=(H);
N=(T),T=(N)->next)

3 使用案例

有人可能會有疑惑,這個雙向鏈表定義如此簡單,只有前驅和后繼兩個指針,甚至連數據域都沒有,那實際該如何使用呢?這個可能就是這組雙向鏈表宏的精妙之處。其在使用過程中并不需要數據域,而是通過指針將結構體串聯成雙向鏈表,并且通過該指針借助 LL_ENTRY宏 能還原出該結構體指針,從而達到操作具體結構體的目的。

如下例子雖然不是完整能跑的程序,但是足夠說明雙向鏈表宏的關鍵用法。程序源碼如下,現對照代碼,描述雙向鏈表宏的大致使用步驟:

定義一個結構體,結構體中必須包含struct llheadlink;雙向鏈表節點,這是后續能通過遍歷雙向鏈表節點,還原出該結構體指針的關鍵;

通過LL_HEAD(listeners);,創建一個雙向鏈表的頭為listeners;

在具體邏輯中,肯定有地方通過LL_TAIL(&listeners, &l->link);或者LL_ADD(H, N),向雙向鏈表的頭listeners添加節點;

在需要操作1.所定義的結構體時,通過LL_FOREACH(&listeners, lp)遍歷出節點指針;

這是最精華的一步,通過4.遍歷出來的節點,傳入宏LL_ENTRY(lp, struct listener, link);中,還原出節點所在的結構體指針,根據邏輯的需要對結構體進行具體相應的操作;

通過宏LL_FOREACH_SAFE來遍歷雙向鏈表,LL_DEL來刪除遍歷出來的節點,達到清空鏈表的作用。

structllhead{
structllhead*prev;
structllhead*next;
};

structlistener{
structllheadlink;
structshttpd_ctx*ctx;/*Contextthatsocketbelongs*/
intsock;/*Listeningsocket*/
intis_ssl;/*ShouldbeSSL-ed*/
};

staticLL_HEAD(listeners);/*Listoflisteningsockets*/

structlistener*l;
LL_TAIL(&listeners,&l->link);

structllhead*lp;
LL_FOREACH(&listeners,lp){
l=LL_ENTRY(lp,structlistener,link);
FD_SET(l->sock,&read_set);
if(l->sock>max_fd)
max_fd=l->sock;
DBG(("FD_SET(%d)(listening)",l->sock));
}

structllhead*lp,*tmp;
LL_FOREACH_SAFE(&listeners,lp,tmp){
l=LL_ENTRY(lp,structlistener,link);
(void)closesocket(l->sock);
LL_DEL(&l->link);
free(l);
}

4 總結

LL_ENTRY(P,T,N)宏是這一組宏的核心,其在具體使用中的功能可以概括為,通過傳入鏈表節點P,還原出節點所在結構體的指針,進而能對結構體進行相應操作

這一組雙向鏈表宏其實形成的是一個循環雙向鏈表;

這些宏最初是極客寫出的,后來在Linux內核中被推廣使用。

原文標題:嵌入式開發中100%會用的幾個宏,建議收藏

文章出處:【微信公眾號:一口Linux】歡迎添加關注!文章轉載請注明出處。

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

    關注

    5092

    文章

    19177

    瀏覽量

    307673
  • 服務器
    +關注

    關注

    12

    文章

    9303

    瀏覽量

    86061
  • 開源框架
    +關注

    關注

    0

    文章

    32

    瀏覽量

    9429
  • 開源代碼
    +關注

    關注

    0

    文章

    36

    瀏覽量

    3009

原文標題:嵌入式開發中100%會用的幾個宏,建議收藏

文章出處:【微信號:yikoulinux,微信公眾號:一口Linux】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    C語言實現動態鏈表的建立

    上期講解了靜態鏈表的實例,但是靜態鏈表建立的節點數量有限,畢竟是手工建立,難免也會出問題, 所以這期講講怎么使用動態的方式建立鏈表,也就是 動態鏈表
    發表于 01-13 15:16 ?1538次閱讀
    C語言實現動態<b class='flag-5'>鏈表</b>的建立

    講解鏈表的刪除、修改以及插入

    上期介紹了動態鏈表怎么建立,以及使用循環的方式怎么輸出整個鏈表各個節點的數據,這期主要講解 鏈表的刪除、修改以及插入 !
    發表于 01-13 15:25 ?2734次閱讀
    <b class='flag-5'>講解鏈表</b>的刪除、修改以及插入

    .h和.c文件的區別到底是什么(精確講解

    必然關系。   這樣你可能會說:啊?那我平時只想調用 xx.c 的某個函數,卻 include了 xx.h文件,豈不是替換后出現了很多無用的聲明?沒錯,確實引入了很多垃圾,但是它卻
    發表于 04-14 16:11

    Linux內核的鏈表操作

    Linux內核的鏈表操作本文詳細分析了 2.6.x 內核鏈表結構的實現,并通過實例對每個鏈表操作接口進行了詳盡的講解。一、
    發表于 08-29 11:13

    C語言玩轉鏈表

    C語言是必學的一個課程,不管你是單片機還是嵌入式物聯網,都是基礎,所以還是要好好學習的今天推薦的資料是關于C語言鏈表的資料我自己看了一下主要說的內容是快速認識數據結構,重點講解鏈表,掌握學習其他數據結構的方法
    發表于 11-13 13:50

    不明白Instaspin-foc的gpio驅動頭文件gpio.h定義

    對于Instaspin-foc的gpio驅動頭文件gpio.h定義,有點不明白,如題:復用寄存器占2BIT,為什么要還要
    發表于 11-20 09:49

    請問2401.h文件引腳的定義在哪里?

    原子哥,為什么在2401.h文件沒有找到關于SCK引腳和MOSI引腳、MISO引腳的定義??改了好多天了,沒找出來~求助求助~是不是時
    發表于 04-30 03:53

    Nano配置文件定義看完你就明白了

    RT-Thread Nano 的配置在 rtconfig.h 中進行,通過開關定義來使能或關閉某些功能,接下來對該配置文件定義進行說
    發表于 03-29 07:19

    stm32f4xx_conf.h文件講解

    stm32f4xx_conf.h文件講解在我的上一篇博客講解到stm32f4xx.h
    發表于 08-24 06:20

    在RT-Thread普通鏈表和侵入式鏈表有何區別

    普通鏈表學習數據結構的時候寫的鏈表是下面這個樣子侵入式鏈表在 RT-Thread 以及 Linux 內核鏈表是這樣定義的在使用的時候是這樣
    發表于 04-11 15:15

    RT-Thread侵入式鏈表的應用有哪些呢

    */struct LNode pre;/ 指向下一個結點 */struct LNode next;/ 指向上一個結點 */}侵入式鏈表在 RT-Thread 以及 Linux 內核鏈表是這樣定義
    發表于 12-05 13:59

    rtdevicd.h這個文件定義是在哪被定義的?

    rtdevicd.h這個文件定義是在哪被定義的
    發表于 02-03 11:49

    了解Linux通用的雙向循環鏈表

    在linux內核,有一種通用的雙向循環鏈表,構成了各種隊列的基礎。鏈表的結構定義和相關函數均在include/linux/list.h
    發表于 05-07 10:44 ?694次閱讀

    linux內核llist.h文件鏈表講解

    鏈表在linux內核、鴻蒙內核、rtos和一些開源代碼中用的非常多。鏈表是雙向鏈表的經典實現方式,總代碼不超過50行,相當精煉。在一些開
    的頭像 發表于 05-23 12:06 ?1939次閱讀

    數組和鏈表在內存的區別 數組和鏈表的優缺點

    數組和鏈表在內存的區別 數組和鏈表的優缺點? 數組和鏈表是常見的數據結構,用于組織和存儲數據。它們在內存的存儲方式以及優缺點方面存在一些
    的頭像 發表于 02-21 11:30 ?1137次閱讀
    主站蜘蛛池模板: 狠狠色狠狠色 | 一区二区三区视频在线观看 | 免费一级特黄特色大片在线观看 | 性夜黄a爽爽免费视频国产 羞羞答答xxdd影院欧美 | www.亚洲成在线 | 一级一黄在线观看视频免费 | 天天摸天天做天天爽天天弄 | 无遮挡很爽很污很黄在线网站 | 久久免费福利视频 | 欧美呜巴又大粗又长 | 日本三级a | www.亚洲免费| 国产一二三区在线 | 色天使在线播放 | 婷婷综合影院 | 四虎影院永久网址 | 天堂网在线资源www最新版 | 老司机午夜永久在线观看 | 狠狠色噜噜狠狠狠 | 色老头综合 | 亚洲欧美视频一区二区三区 | 好吊妞视频988在线播放 | 亚洲情a成黄在线观看 | 五月天婷婷丁香 | 嘿嘿午夜| 四虎影院永久地址 | 中文天堂在线最新2022更新 | 91大神在线精品网址 | 中文字幕亚洲色图 | 久久激情网 | 日本免费色视频 | 伊人久久亚洲综合 | a级毛片免费观看网站 | 免费视频国产 | 久久精品美女久久 | 亚洲综合激情另类专区 | 日韩免费三级电影 | 亚洲最大成人 | 2020夜夜操| 老师喂我吃她的奶水脱她胸罩 | 国产男女免费视频 |