结构体内存大小

news2024/11/24 20:08:25

000、前言

要想计算结构体内存大小,就会涉及到一个结构体内存对齐的问题,而不是对其成员进行简单的加运算

(1)在写本博客之前

有位同学和我讨论了一个学校的题目,题目如下:
在这里插入图片描述
我借这道题目问了另外一位同学,这位同学认为答案为12,理由是这样的

“4(int的大小)+4(y数组的大小)+4(float的大小)=12”

(2)我的思考

老实说,我不太好评价学校的C语言课,但是这道题目如果没有细讲的话,很有可能造成一定程度上的误解。稍微知道结构体内存对齐的老铁就会明白,这个结构体的大小的确是12,但是绝不是简单意义上的大小,言外之意:这道题是个大坑!

(3)我的代码提问

我当时写了一个文档发给了这位同学

int main()
{
    struct a
    {
        int x;
        char y[4];
        float z;
    };
    //一种朴素的想法当然是4+4+4啦,但是肯定不对……
    //不信你看这个程序
    printf("%zd\n", sizeof(struct a));
    //输出的也是12啊,怎么不对了?是的,不对,这个只是巧合,因为成员刚好大小都是4个字节,这与放了三个int成员是没有太大区别
    //但是如果是不同的就涉及到结构体内存对齐的问题了
    //假设我们将结构体内部成员换一下,存在一个char类型的成员
    struct b
    {
        char i;
        int x;
        char y[4];
        float z;
    };
    //按照直接加的逻辑,就是1+4+4+4==13
    printf("%zd\n", sizeof(struct b));
    //你会发现结果是16!
    //如果可以系统了解一下结构体对齐的知识吧,需要的话我可以发个链接给你
    //但是这道题由于成员设置的比较简单(因为结构体成员的类型都是4个字节)直接简单相加得到的结果确实是对的,都是这样的理解方式是有问题的!!
    return 0;
}

001、结构体内存对齐规则

(1)规则一:第一个成员变量在结构体变量偏移量为0的地址处

(2)规则二:其他成员变量要对齐到“对齐数”的整数倍的地址处(每个成员拥有一个对齐数,对齐数==“编译器默认的一个对齐数(VS默认这个数是8)”与“该成员大小”的较小值)

(3)规则三:结构体的总大小为最大对齐数(每个成员对齐数最大的)整数倍处

(4)规则四:如果嵌套了结构体,先计算这个被内嵌的结构体大小,然后将这个被内嵌的结构体整体当成一个新类型就行

002、为什么存在内存对齐?

(1)性能原因

数据结构应该尽可能的在自然边界上对齐。如果处理器访问了没有对齐的内存,可能需要多次访问内存空间才能得到一个完整的数据

(2)平台原因

不是所有平台都可以访问任意地址上的数据的,某些硬件平台只能在某些地址取出某些特定类型的数据,否则抛出硬件异常

(3)实质原因

牺牲空间换时间的做法

003、结构体大小推导例子

(1)例子一

struct S1
{
    char c1;//char大小为1,和默认对齐数8比,该成员的成员对齐数是1
    char c2;//char大小为1,和默认对齐数8比,该成员的成员对齐数是1
    int i;//整型大小为4,和默认对齐数8比,该成员的成员对齐数是4
};//最大成员对齐数是4,结构体总大小必须是4的倍数
printf("%d\n", sizeof(struct S1));//结果为8

在这里插入图片描述

(2)例子二

struct S2
{
    char c1;//大小为1,和默认对齐数比,该成员的成员对齐数是1
    int i;//大小为4,和默认对齐数比,该成员的成员对齐数是4
    char c2;//大小为1,和默认对齐数比,该成员的成员对齐数为1
};//最大成员对齐数是4,结构体的总大小必须是4的倍数
printf("%d\n", sizeof(struct S2));//结果为12,注意不是9,因为要满足规则三

在这里插入图片描述

(3)例子三

//嵌套结构体
struct S3
{
    double d;//大小为8,和默认对齐数8相比,该成员的成员对齐数为8
    char c;//大小为1,与默认对齐数8相比,该成员的成员对齐数为1
    int i;//大小为4,与默认对齐数8相比,该成员的成员对齐数为4
};//最大成员对齐数为8,结构体的总大小必须是8的倍数
printf("%d\n", sizeof(struct S3));//输出16

struct S4
{
    char c1;//大小为1,与默认对齐数8相比,该成员的成员对齐数为1
    struct S3 s3;//大小为16,与默认对齐数8相比,该成员的成员对齐数为8
    double d;//大小为8,与默认对齐数8相比,该成员的成员对齐数为8
};//最大成员对齐数为8,结构体的总大小必须是8的倍数
printf("%d\n", sizeof(struct S4));//值为32

S3的大小
在这里插入图片描述
S4的大小
在这里插入图片描述

004、在VS2022中如何修改默认对齐数?

(1)使用#pragma

利用#pragma这个预处理指令就可以改变默认对齐数

(2)具体代码

使用#pragma的具体代码(这里的代码可以自己尝试画图解决)

①代码一

#include <stdio.h>
#pragma pack(8)//把默认对齐数设置为8
struct S1
{
    char c1;//char大小为1,和设置后的默认对齐数8比,该成员的成员对齐数是1
    char c2;//char大小为1,和设置后的默认对齐数8比,该成员的成员对齐数是1
    int i;//整型大小为4,和设置后的默认对齐数8比,该成员的成员对齐数是4
};//最大成员对齐数是4,结构体大小必须是4的倍数
#pragma pack()//取消设置的默认对齐数,还原为默认值

int main()
{
    printf("%d\n", sizeof(struct S1));//结果为8
}

②代码二

#pragma pack(1)//设置默认对齐数为1
struct S2
{
    char c1;//大小为1,和设置后的默认对齐数1比,该成员的成员对齐数是1
    int i;//大小为4,和设置后的默认对齐数1比,该成员的成员对齐数是1
    char c2;//大小为1,和设置后的默认对齐数1比,该成员的成员对齐数为1
};//最大成员对齐数是1,结构体大小必须是1的倍数
#pragma pack()//取消设置的默认对齐数,还原为默认

int main()
{
    printf("%d\n", sizeof(struct S2));//结果为6
    return 0;
}

这里注意,当默认对齐数为1的时候,此时对结构体成员进行简单加法运算得到结构体大小是没有问题的

005、注意

如果你认真看完了上面的内容,你就会发现这里面最重要的就是对齐数的确认,而编译器自身携带的默认对齐数更是会直接影响结构体的的大小。

然而现实是,不同的编译器的默认对齐数有可能是不一样的,甚至有的编译器直接就没有这方面的规定!!!例如下面这一串代码

#include <stdio.h>
int main()
{
    //嵌套结构体
    struct S3
    {
        double d;
        char c;
        long double i;
    };
    printf("%d\n", sizeof(struct S3));

    struct S4
    {
        char c1;
        struct S3 s3;
        double d;
    };
    printf("%d\n", sizeof(struct S4));
    return 0;
}

(1)VS环境下

在这里插入图片描述

(2)在vscode使用mingw64的gcc工具环境下

在这里插入图片描述

据听说,本题还被作为考试题目,当作检验C语言课程学习来使用???如果讲了也就罢了,若是没讲,就算考生得到12这个结果(指开头的题目),真的不会影响他对结构体内存的理解么?私认为,本题目不够严谨……

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/403654.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

JS学习笔记day05(完结)!

今日内容 零、 复习昨日 一、作业 二、BOM 三、定时器 四、正则表达式 零、 复习昨日 事件 事件绑定方式鼠标事件 onmouseoveronmouseoutonmousemove 键盘事件 onkeydownonkeyuponkeypress 表单事件 onfocusonbluronchangeonsubmit 页面加载事件 onload dom dom树查找dom do…

what data contract

URLS 笔记内容主要来自 https://learn.microsoft.com/en-us/azure/cloud-adoption-framework/scenarios/cloud-scale-analytics/architectures/data-contracts Data contracts Data contracts are similar to service contracts or data delivery contracts.A contract also…

One-YOLOv5 v1.2.0发布:支持分类、检测、实例分割

One-YOLOv5 v1.2.0正式发布。完整更新列表请查看链接&#xff1a;https://github.com/Oneflow-Inc/one-yolov5/releases/tag/v1.2.0&#xff0c;欢迎体验新版本&#xff0c;期待你的反馈。 1 新版本特性 1. 同步了Ultralytics YOLOv5的上游分支v7.0&#xff0c;同时支持分类、目…

多校园SaaS运营智慧校园云平台源码 智慧校园移动小程序源码

智慧校园管理平台源码 智慧校园云平台源码 智慧校园全套源码包含&#xff1a;电子班牌管理系统、成绩管理系统、考勤人脸刷卡管理系统、综合素养评价系统、请假管理系统、电子班牌发布系统、校务管理系统、小程序移动端、教师后台管理系统、SaaS运营云平台&#xff08;支持多学…

每天学一点之Stream流相关操作

StreamAPI 一、Stream特点 Stream是数据渠道&#xff0c;用于操作数据源&#xff08;集合、数组等&#xff09;所生成的元素序列。“集合讲的是数据&#xff0c;负责存储数据&#xff0c;Stream流讲的是计算&#xff0c;负责处理数据&#xff01;” 注意&#xff1a; ①Str…

Java面试总结(六)

进程和线程的区别 根本区别&#xff1a; 进程时操作系统资源分配的基本单位&#xff0c;而线程是处理器任务调度和执行的基本单位。 资源开销&#xff1a; 每个进程都有自己独立的代码和数据空间&#xff08;程序上下文&#xff09;&#xff0c;进程之间的切换开销比较大&…

狂扫近300万读者,蟒蛇书升级版即将出版,招募审读人

狂扫全世界近 300 万爱好者成为编程领域的现象级爆品豆瓣累计超过 5000 人评价第2版中文版获得了 9.3 分的好评Amazon 近 10000 人评价第2版原版获得了 4.7 星好评毫不夸张&#xff0c;它是全世界读者心中的 Python 入门圣经因为封面上是一只蠢萌的蟒蛇这本书又被读者亲切地称为…

SQL注入——文件上传

目录 一&#xff0c;mysql文件上传要点 二&#xff0c;文件上传指令 一句话木马 三&#xff0c;实例 1&#xff0c;判断注入方式 2&#xff0c;测试目标网站的闭合方式&#xff1a; 3&#xff0c;写入一句话木马 4&#xff0c;拿到控制权 一&#xff0c;mysql文件上传…

【面试系列】volatile的底层原理

并发编程的三大特性 原子性可见性原子性 JAVA内存模型 Java内存模型&#xff08;Java Memory Model&#xff09;主要分为主内存和线程工作内存。 主内存&#xff1a;方法区和堆空间 线程工作内存&#xff1a;虚拟机栈&#xff0c;本地方法栈&#xff0c;程序计数器。 所有…

centos安装docker详细步骤

目录 一.前言 1.环境要求2.官网中文安装参考手册 二.安装步骤 1.卸载旧版本2.安装需要的软件包3.设置docker镜像源 1.配置docker镜像源 方式1&#xff1a;官网地址(外国)&#xff1a;方式2&#xff1a;阿里云源&#xff1a;2.查看配置是否成功 4.更新yum软件包索引5.可以查看…

C++ | 探究函数重载的原理:函数名修饰【基于Windows + Linux双系统】

文章目录一、前言【中国乒乓和中国男足】【文言文一词多义】二、函数重载概念引入1、参数【类型】不同构成重载2、参数【个数】不同构成重载3、参数【类型顺序】不同构成重载三、函数重载的原理1、回顾程序编译 链接的过程2、Linux下【objdump】查看反汇编3、Windows下反汇编查…

Maven安装与配置,IDEA配置Maven

文章目录1. 安装本地Maven2. 安装3. 配置环境变量4. 配置settings.xml文件5. IDEA配置1. 安装本地Maven 官网下载&#xff1a;https://maven.apache.org/download.cgi 2. 安装 把下载好的maven压缩包解压到一个没有中文&#xff0c;空格或其他特殊字符的文件夹&#xff0c;…

数据结构与算法基础(王卓)(15):KMP算法详解(含速成套路和详细思路剖析)

如果时间不够&#xff0c;急&#xff08;忙&#xff09;着应付考试没心思看&#xff0c;直接参考&#xff08;照抄&#xff09;如下套路&#xff1a; PART 1&#xff1a;关于next [ j ] PPT&#xff1a;P30 根据书上以及视频上给出的思路&#xff08;提醒&#xff09;&#x…

JSONP劫持

注意&#xff1a;仅用于技术讨论&#xff0c;切勿用于其他用途&#xff0c;一切后果与本人无关&#xff01;&#xff01;&#xff01; 个人博客地址&#xff1a;HJW个人博客 扩展&#xff1a; 面试问题&#xff1a;CSRF的两种方法中&#xff0c;CORS和JSONP的区别&#xff1…

dubbo进阶——服务导出

服务导出 在这里记录一下对" Dubbo 导出服务的过程"的研究。 触发时机 public class ServiceBean<T> extends ServiceConfig<T> implements InitializingBean, DisposableBean, ApplicationContextAware, ApplicationListener<ContextRefreshedEv…

代码随想录--数组--螺旋矩阵题型

n等于几&#xff0c;螺旋矩阵最外层行列数就等于几。如n等于3所以最外层一圈的行、列都是3个数字&#xff0c;如左图&#xff0c;n等于4所以最外层的行、列都是4个数字&#xff0c;如右图&#xff1a; 题目要求我们输出(以n3为例)[[123][894][765]]&#xff0c;题目只是给我们一…

MySQL:索引与事物

目录 简单了解索引的底层数据结构 索引的概念&#xff1a; 索引存在的意义&#xff1a; 索引的使用&#xff1a; 索引实现的数据结构 B树 B 树 B 树的特点 B 树的优势 事物 事物的概念 事物的使用 事物的四大特性 并发可能引起的问题 脏读问题 不可重复读 幻读…

未系安全带识别系统 yolo

未系安全带识别系统通过pythonyolo智能视频分析技术&#xff0c;未系安全带识别算法对画面中高空作业人员未系安全带行为进行监测&#xff0c;未系安全带识别算法监测到人员未穿戴安全带时&#xff0c;立即通知后台人员及时处理触发告警。Yolo算法采用一个单独的CNN模型实现end…

算法24:LeetCode_并查集相关算法

目录 题目一&#xff1a;力扣547题&#xff0c;求省份数量 题目二&#xff1a;岛屿数量 题目三&#xff1a;岛屿数量拓展 什么是并查集&#xff0c;举个简单的例子。学生考试通常会以60分为及格分数&#xff0c;我们将60分及以上的人归类为及格学生&#xff0c;而60分以下归…

mysql学习之数据系统概述

☀️马上要成为打工人&#xff0c;这几天把前面的知识都捡了捡&#xff0c;发现自己对关系数据库这块的学习还有所缺失&#xff0c;于是本章开始学习mysql 这里写目录标题1. 数据库系统的发展1.1 人工管理阶段1.2 文件系统阶段1.3 数据库阶段1.4 大数据阶段2 数据库系统的组成2…