C语言 | 如何巧妙地求结构体成员的大小?

1024G 嵌入式资源大放送!包括但不限于C/C++、单片机、Linux等。关注微信公众号【嵌入式大杂烩】,回复1024,即可免费获取!

在C语言编程中,有时候需要知道某结构体中某成员的大小,比如使用堆内存来存储结构体中的某成员时,需要知道该成员的大小,才好确定所需申请的空间大小。

求某结构体中某成员的大小,你会怎么做?

例子:

typedef struct
{
 char a;  
 char c; 
 short b;         
 int d;
 char e;
}test_struct;

求 d 成员所占内存空间的大小。

方法一

我们可以先定义一个结构体变量,然后再使用sizeof求出。

#include <stdio.h>

typedef struct
{
 char a;  
 char c; 
 short b;         
 int d;
 char e;
}test_struct;

int main(void)
{
 test_struct test_s;  
 printf("sizeof(test_s.d) = %d\n", sizeof(test_s.d));
 return 0;
}

运行结果:

但是我们为了得到一个成员的大小,而专门定义一个结构体变量,而这个变量也没有其它的用途,有点浪费资源,或者说这种方法low了。

方法二

肉眼观察法。。。。。

比如在32bit环境下,我们一眼看出d是int类型,就是4个字节。然后想咋用就咋用。这个简单就不讨论了。

方法三

装逼法。。。

代码:

#include <stdio.h>

typedef struct
{
 char a;  
 char c; 
 short b;         
 int d;
 char e;
}test_struct;

int main(void)
{ 
 printf("sizeof(((test_struct*)0)->d) = %d\n", sizeof(((test_struct*)0)->d));
 printf("sizeof(((test_struct*)0)->a) = %d\n", sizeof(((test_struct*)0)->a));
 printf("sizeof(((test_struct*)0)->b) = %d\n", sizeof(((test_struct*)0)->b));
 printf("sizeof(((test_struct*)0)->c) = %d\n", sizeof(((test_struct*)0)->c));
 return 0;
}

运行结果:

类似((test_struct*)0)->d这样的用法是个固定用法,把0地址转换为test_struct结构的指针,对于结构体指针,使用->符号就是取其成员,再使用sizeof就可以求得其大小。

这里不一定是0地址,其它地址也可以,但一般都会使用0地址。这种方法较方法一的好处就是不用定义一个多余的变量。

这种方法很重要,需要掌握,可能你平时编程不会使用这种方法,但这种方法很重要。在很多优秀的代码中会出现类似形式的宏代码,例如:

上例可封装一个宏定义:

#define  MEM_SIZE(type, member)   sizeof(((type*)0)->member)

求某成员在结构体中的偏移量:

#define  OFFSETOF(type, member)  ( (size_t)( &( ( (type*)0 )->member ) ) )

求结构体偏移量在C语言头文件中stddef.h也有提供,使用方法如:

// 微信公众号:嵌入式大杂烩
#include <stdio.h>
#include <stddef.h>

#define  OFFSETOF(type, member)  ( (size_t)( &( ( (type*)0 )->member ) ) )

typedef struct
{
 char a;  
 char c; 
 short b;         
 int d;
 char e;
}test_struct;

int main(void)
{  
 /* stddef.h宏 */
 printf("offset(a): %d\n", offsetof(test_struct, a));
 printf("offset(c): %d\n", offsetof(test_struct, c));
 printf("offset(b): %d\n", offsetof(test_struct, b));
 printf("offset(d): %d\n", offsetof(test_struct, d));
 printf("offset(e): %d\n", offsetof(test_struct, e));
 
 /* 自定义宏 */
 printf("OFFSETOF(a): %d\n", OFFSETOF(test_struct, a));
 printf("OFFSETOF(c): %d\n", OFFSETOF(test_struct, c));
 printf("OFFSETOF(b): %d\n", OFFSETOF(test_struct, b));
 printf("OFFSETOF(d): %d\n", OFFSETOF(test_struct, d));
 printf("OFFSETOF(e): %d\n", OFFSETOF(test_struct, e));
 
 return 0; 
}

运行结果:

使用这个求结构体偏移量的宏我们就可以很好地知道结构体成员的在内存中的存储情况。关于结构体内存对齐,可移步至往期笔记:结构体内存对齐你真的弄懂了?

以上就是本次分享的求结构体成员的三种方法。重点掌握方法三,因为在很多优秀的代码中都有使用到类似的方法。

猜你喜欢

sizeof你真的弄明白了吗?来看看这个例子

一种简单、实用的测量程序运行时间的方法

请写一个函数输出如下波形

1024G 嵌入式资源大放送!包括但不限于C/C++、单片机、Linux等。关注微信公众号【嵌入式大杂烩】,回复1024,即可免费获取!

  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入式大杂烩

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值