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】欢迎添加关注!文章转载请注明出处。

审核编辑:彭静
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 嵌入式
    +关注

    关注

    5082

    文章

    19104

    浏览量

    304808
  • 服务器
    +关注

    关注

    12

    文章

    9123

    浏览量

    85324
  • 开源框架
    +关注

    关注

    0

    文章

    32

    浏览量

    9412
  • 开源代码
    +关注

    关注

    0

    文章

    36

    浏览量

    2939

原文标题:嵌入式开发中100%会用的几个宏,建议收藏

文章出处:【微信号:yikoulinux,微信公众号:一口Linux】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    C语言实现动态链表的建立

    上期讲解了静态链表的实例,但是静态链表建立的节点数量有限,毕竟是手工建立,难免也会出问题, 所以这期讲讲怎么使用动态的方式建立链表,也就是 动态链表
    发表于 01-13 15:16 1439次阅读
    C语言实现动态<b class='flag-5'>链表</b>的建立

    讲解链表的删除、修改以及插入

    上期介绍了动态链表怎么建立,以及使用循环的方式怎么输出整个链表各个节点的数据,这期主要讲解 链表的删除、修改以及插入 !
    发表于 01-13 15:25 2554次阅读
    <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 672次阅读

    linux内核llist.h文件链表讲解

    链表在linux内核、鸿蒙内核、rtos和一些开源代码中用的非常多。链表是双向链表的经典实现方式,总代码不超过50行,相当精炼。在一些开
    的头像 发表于 05-23 12:06 1879次阅读

    数组和链表在内存的区别 数组和链表的优缺点

    数组和链表在内存的区别 数组和链表的优缺点  数组和链表是常见的数据结构,用于组织和存储数据。它们在内存的存储方式以及优缺点方面存在一些
    的头像 发表于 02-21 11:30 1021次阅读