0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

详细聊聊container_of这个宏定义

lilihe92 来源:最后一个bug 2023-11-13 09:45 次阅读

正文

大家周末好,我是bug菌~ 今天主要是跟大家详细聊聊container_of这个宏定义,非常经典的宏,只是一直没有抽时间细细品味,今天就跟大家一起来看看有何神奇之处:

1offsetof

首先我们需要简单看看offsetof(TYPE, MEMBER) 这个宏定义,它是用于计算一个结构体中某个成员的偏移量。

其第一个参数 TYPE 是一个结构体类型,第二个参数 MEMBER 是 TYPE 中的一个成员变量名。

它将返回类型为 size_t 的整数,表示 MEMBER 相对于 TYPE 起始地址的偏移量。

基本原理是根据 C 语言的数据对齐机制,成员变量在类型定义中的相对位置决定了它的偏移量。

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

该宏定义使用了C语言中的指针运算和类型转换。具体实现步骤如下:

1、(TYPE *)0:将0强制类型转换为指向类型为TYPE的指针,得到了一个结构体TYPE的空指针。

2、&((TYPE *)0)->MEMBER:求出结构体类型TYPE中成员MEMBER的地址。其巧妙之处在于,由于空指针不指向任何对象,因此这个成员的地址就是相对于结构体首地址的偏移量。

3、(size_t):将偏移量转换为无符号整型数,以满足C语言标准库中对offsetof()返回值的类型要求。

该宏定义可以在编译时就直接计算出偏移量,避免了运行时的计算开销,因此比通过变量名访问成员的方式更为高效,通常用在需要直接访问结构体成员的底层代码中,例如在操作系统内核、嵌入式系统以及一些高性能计算应用中。

structTestStruct{
intvalue1;
charvalue2;
doublevalue3;
};

size_toffset=offsetof(structTestStruct,value2);

如上例,offset 变量将会存储 value2 相对于 TestStruct 起始地址的偏移量。在这种情况下,因为 TestStruct 中的 value1 通常占用了 4 个字节,value2 占用了 1 个字节,所以 value2 相对于结构体起始地址的偏移量应该是 4。

2container_of

讲完offsetof,来到今天的主角container_of,container_of()是一个在linux内核中经常使用的宏,用于获取一个结构体成员指针所在它所属的结构体的指针,有点绕口,细细品味。

该宏包括也主要包括三个参数:

ptr:结构体中某个成员的指针;

type:结构体类型名称;

member:结构体中ptr指向的成员名称。

首先,宏container_of()确定了ptr指向的成员在结构体中的偏移(offset)。通过offsetof()宏就可以得到这个偏移,其参数为结构体类型和成员名称。得到偏移后,再通过减去偏移的方式得到指向整个结构体的指针,巧妙吧。

具体实现如下:

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

其中,typeof是GCC的一个扩展关键字,用于返回一个表达式的类型,可惜,大部分非GCC编译器不一定能支持。

假设ptr指向的成员变量的类型为T,__mptr就是一个指向T类型的指针。然后,调用offsetof()即可得到member在type类型中的偏移量,最后返回一个指向type类型的指针。

注意,尖括号不能省略,因为它表示类型转换。此外,container_of()宏使用了一个GCC的语言扩展"statement expression",即后面的{},可以在其中包含多条语句。

下面给出一个示例,用于说明container_of()的使用方法:

#include
#include

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

structstudent{
intid;
charname[20];
};

intmain(){
structstudentstu={10001,"ZhangSan"};
char*pname=stu.name;
structstudent*pstu=container_of(pname,structstudent,name);

printf("ID:%d,Name:%s
",pstu->id,pstu->name);
return0;
}

如上例,pname指向stu的name成员,通过container_of()宏获得了指向整个struct student结构体的指针pstu,然后就可以访问id和name成员了。


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

    关注

    180

    文章

    7614

    浏览量

    137619
  • 结构体
    +关注

    关注

    1

    文章

    130

    浏览量

    10869
  • 宏定义
    +关注

    关注

    0

    文章

    51

    浏览量

    9048

原文标题:搞懂它,就可以把结构体玩活了~

文章出处:【微信号:最后一个bug,微信公众号:最后一个bug】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    C语言定义使用技巧

    写好C语言,漂亮的定义很重要,使用定义可以防止出错,提高可移植性,可读性,方便性等等。下面列举一些成熟软件中常用的
    发表于 07-29 09:35 1132次阅读

    Linux内核中container_of原理详解

    Linux内核中经常可见container_of的身影,它在实际驱动的编写中也是广泛应用。
    发表于 07-14 15:19 345次阅读
    Linux内核中<b class='flag-5'>container_of</b>原理详解

    这个奇怪的定义要干啥???

    #define start_timer_int_ms(count) \{ \TA1CCTL0 = CCIE;\TA1CCR0 = count*5; \TA1CTL = TASSEL_1 + MC_1 + TACLR + ID__8; \}这个奇怪的
    发表于 04-20 19:22

    揭开linux内核中container_of的神秘面纱

    在linux 内核中有一个大名鼎鼎的container_of(),这个是用来干嘛的呢?我们先来看看它在内核中是怎样定义的。呵呵,乍一看不
    发表于 11-11 10:02

    offsetofcontainer_of详解

    // 元素元素名// 这个返回的就是指向这个结构体变量的指针,类型是(type *)#define container_of(ptr, type, member) ({\const
    发表于 10-13 16:35

    定义问题!

    各位,请问一下,这个定义代表什么意思?#define OUTPUT_WAVE(pointer, d)PORTC = pointer[d*N_PORTS + 0] 谢谢
    发表于 03-12 10:30

    rtdevicd.h这个文件中的定义是在哪被定义的?

    rtdevicd.h这个文件中的定义是在哪被定义
    发表于 02-03 11:49

    C语言定义使用技巧

    写好C语言,漂亮的定义很重要,使用定义可以防止出错,提高可移植性,可读性,方便性 等等。下面列举一些成熟软件中常用得
    发表于 11-13 12:04 36次下载

    内联函数和定义的区别介绍

    定义是C语言提供的三种预处理功能的其中一种,这三种预处理包括:定义、文件包含、条件编译。定义
    发表于 12-15 15:33 2354次阅读
    内联函数和<b class='flag-5'>宏</b><b class='flag-5'>定义</b>的区别介绍

    不带参数的定义是什么?不带参数的定义的资料介绍详细过程概述

    c语言中有一个定义,其中有一类就是不带参数的定义定义是C提供的三种预处理功能的其中一
    发表于 09-04 15:38 5次下载

    使用AVR单片机编写的定义的应用代码详细资料免费下载

    本文档的主要内容详细介绍的使用AVR单片机编写的定义的应用代码详细资料免费下载。
    发表于 11-12 08:00 3次下载
    使用AVR单片机编写的<b class='flag-5'>宏</b><b class='flag-5'>定义</b>的应用代码<b class='flag-5'>详细</b>资料免费下载

    对于定义与重复定义的问题

    对于定义与重复定义的问题
    发表于 11-29 18:21 1次下载
    对于<b class='flag-5'>宏</b><b class='flag-5'>定义</b>与重复<b class='flag-5'>定义</b>的问题

    Linux内核中的/container_of分析

    今天在看平台设备实现的时候,看到to_xxx开头的“函数”。包括在内核中也有很多此类的“函数”,其实他们都是container_of。因为内核是链表和结构体的世界,因此内核中有大量需求要 根据结构体成员获取结构体地址 ,或者根据结构体类型和结构体成员类型获取成员在结构
    发表于 06-23 14:26 442次阅读
    Linux内核中的<b class='flag-5'>宏</b>/<b class='flag-5'>container_of</b>分析

    c语言带参数的定义

    c语言带参数的定义  C语言定义是一种替换机制,它可以将一个标识符替换为一个代码片段。
    的头像 发表于 09-04 17:45 2488次阅读

    container_of(),太妙了~

    的作用是通过结构体成员的地址和结构体类型推导出结构体的地址,type是指结构体的类型,member是成员在结构体中的名字,ptr是该成员在type结构体中的地址。
    的头像 发表于 09-10 10:59 534次阅读
    <b class='flag-5'>container_of</b>()<b class='flag-5'>宏</b>,太妙了~