在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个字节,使用sizeof(int)求出。然后想咋用就咋用。这个简单就不讨论了。

方法三

装X法。。。

代码:

#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;   

}

运行结果:

 

使用这个求结构体偏移量的宏我们就可以很好地知道结构体成员的在内存中的存储情况。

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

如有错误,欢迎指出!谢谢~

看到这里是不是又学到了很多新知识呢~

如果你很想学编程,小编推荐我的C语言/C++编程学习基地【点击进入】!

都是学编程小伙伴们,带你入个门还是简简单单啦,一起学习,一起加油~

还有许多学习资料和视频,相信你会喜欢的!

涉及:游戏开发、常用软件开发、编程基础知识、课程设计、黑客等等......

 

 

【C语言】这种求结构体成员大小的方法,你可能需要了解一下~的更多相关文章

  1. 指针直接赋值为整型AND利用宏定义求结构体成员偏移量

    首先我们要更正一个很熟悉的概念,那就是指针不仅仅是“地址”,指针还有一个很重要的特性,那就是“类型”. 指针初始化时,“=”的右操作数; 除外,该语句表示指针为空): 所以 ; 这样的代码是不允许的. ...

  2. 怎样求结构体成员的偏移地址 || 结构体的 sizeof 总结

    C 语言中同意将值为 0 的变量强制转换成任一类型的指针,转换结果是一个NULL指针. (type*)0 // 一个 type 类型的NULL指针 用这个指针訪问结构体内的成员是非法的,可是 & ...

  3. C语言中访问结构体成员时用‘.’和‘->’的区别

    举个例子,定义了一个叫Student,别名为stu的结构类型,我们声明了一个结构体变量叫stu1,声明了一个结构体指针为stuP. typedef struct Student { char name ...

  4. go语言基础之结构体成员的使用指针变量

    1.结构体成员的使用:指针变量 示例: package main //必须有个main包 import "fmt" //定义一个结构体类型 type Student struct ...

  5. go语言基础之结构体成员的使用普通变量

    1.结构体成员的使用普通变量 示例: package main //必须有个main包 import "fmt" //定义一个结构体类型 type Student struct { ...

  6. C语言:存取结构体成员的点运算符(.)和箭头运算符(->)的区别

    转自:http://blog.csdn.net/taric_ma/article/details/7397362 一直以为这两个是没有什么区别的,可以相互替换,今天又翻了一下<C语言核心技术&g ...

  7. offset求结构体成员的偏移量

    [代码]  C++ Code  12345678910111213141516171819202122232425262728293031   /* version: 1.0 author: hell ...

  8. struct的成员对齐问题-结构体实际大小问题

    struct的成员对齐 注意:为了方便说明,等号左边是每个数据单独所占长度,右边是最终空间大小,以字节为单位. 一.什么时间存在对其问题:(32位机对齐方式是按照4字节对其的,以下所有试验都是在32位 ...

  9. 关于C语言中结构体中的结构体成员导致的字节对齐问题

    关于结构体的字节对齐是什么,就不赘述,再此附上一篇文章,介绍字节对齐:http://www.linuxsong.org/2010/09/c-byte-alignment/ 这里的结构体字节对齐的数据类 ...

随机推荐

  1. i春秋公益赛之BFnote

    题目链接:https://buuoj.cn/challenges#gyctf_2020_bfnote 首先检查程序开的保护: 发现程序只开了canary和NX保护,接下来查看IDA反汇编出来的为代码, ...

  2. JS语法_类型

    类型 JS 的数据类型 boolean number string undefined null symbol object TS 额外的数据类型 void BigInt 是一种内置对象,它提供了一种 ...

  3. shell数组的用法

    在shell里面想获取某个变量的值,使用$符开头,如:$a或者${a}即可. 获取数组长度 arr_length=${#arr_number[*]}或${#arr_number[@]}均可,即形式:$ ...

  4. pytest(1):安装与使用

    pytest介绍 1.pytest是比较成熟的一款python测试框架 2.简单灵活,容易上手,对比unitest框架更丰富. 3.单元测试和复杂的功能测试,都支持. 4.同时支持selenium,a ...

  5. CTF-BugKu-加密

    2020.09.12 恕我直言,上午做WeChall那个做自闭了,下午复习一下之前做过的. 做题 第一题 滴答~滴 https://ctf.bugku.com/challenges#滴答~滴 摩斯密码 ...

  6. linux如何把普通用户添加到sudo组

    sudo原理:运行命令时,系统检查/etc/sudoers 配置文件,看这个用户是否有执行sudo的权限,如果有权限,系统要求输入用户自己的密码,如果密码输入正确,系统会以root身份运行 passw ...

  7. [算法与数据结构]使用Java泛型实现栈

    ###题解 1 实现内部类node 2 维护top为头节点的链表 3 操作 操作1:push() 操作2: pop() 操作3: isEmpty() ###代码 package Exam; class ...

  8. spring mvc(1) 为什么要使用mvc

    在使用spring mvc之前,我们首先要理解我们为什么要使用spring mvc.关于这个问题我们可以看一下java web的简单发展过程. 1. servlet 开发阶段 上世纪90年代,随着In ...

  9. 《Java并发编程的艺术》笔记

    第1章 并发编程的挑战 1.1 上下文切换 CPU通过时间片分配算法来循环执行任务,任务从保存到再加载的过程就是一次上下文切换. 减少上下文切换的方法有4种:无锁并发编程.CAS算法.使用最少线程.使 ...

  10. python os模块方法详解

    os.access() 方法使用当前的uid/gid尝试访问路径.大部分操作使用有效的 uid/gid, 因此运行环境可以在 suid/sgid 环境尝试. 实例: os.chdir() 方法用于改变 ...