计算结构体的大小(结构体的内存对齐)

一:问题

问题所在:两个结构体应该都是6个字节大小,为什么一个12,一个6???

二:如何正确的计算结构体大小?
首先得掌握结构体的对齐规则:
第一: 第一个成员在与结构体变量偏移量为 0 的地址处。

解释:      

        1,假设一个格子占一个格子,则右边的数字为偏移量。

        2,假设这是一个结构体内部的内存,储存结构体内部的数据从第一格开始,那么右边的数就是与结构体变量的偏移量。

        3,根据对齐规则的第一点,所以结构体S1第的一个值c1应该放在如图所示红色的位置,占一个字节。

第二: 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处( 图中右边的数字 )。
对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值
编译器默认的一个对齐数 :
VS中默认的值为8
Linux中没有默认对齐数,对齐数就是成员自身的大小
我们是在vs中,所以默认的对齐数是8。

解释:

        1,所以经过第二点,我们可知内存中c1和c2以及i的存放位置了 。

        2,i本身大小是4,而vs默认对齐数是8,而对齐数是默认的一个对齐数 与 该成员大小的较小值,所以对齐数为4,所以 i 就要对齐到4的整数倍的地址处(图中右边的数字)。所以最近的4的整数倍也就是如图所示的蓝色第一格,刚好是4的一倍

        3,c2本身大小为1,而vs默认对齐数是8,所以1和8,进行比较,较小值为1,所以对齐数为1,那c2就要去找1的倍数处的地址(图中右边的数字),任何一个数都是1的倍数,所以就选择了紧接着i的红色格子。

第三:结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。 

解释:

因为结构体总体的大小是最大对齐数(每个成员变量都有一个对齐数)的整数倍。而c1和c2和i的对齐数中,最大的是4,所以结构体大小应该是4的倍数,因为此时结构体已经9个字节了,最近的符合规则的就是12,所以又往下浪费了3个字节。 总打下就是12个字节。

第四:  如果嵌套了结构体的情况,
嵌套的结构体这个整体要对齐到自己的最大对齐数(也就是它内部元素的最大对齐数)的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。
三:为什么存在内存对齐 ?
1. 平台原因(移植原因):
不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特
定类型的数据,否则抛出硬件异常。
2. 性能原因:
数据结构(尤其是栈)应该尽可能地在自然边界上对齐。
原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访
问。
总体来说: 结构体的内存对齐是拿空间来换取时间的做法。
所以:那在设计结构体的时候,我们既要满足对齐,又要节省空间,如何做到:
让占用空间小的成员尽量集中在一起。
比如上图的S1S2类型的成员一模一样,但是S1S2所占空间的大小有了一些区别。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Head First Design Patterns -适配器模式与外观模式

适配器模式 什么是适配器模式 适配器模式,将一个类的接口转换成客户期望的另一个接口。适配器让原本接口不兼容的类可以合作。 类图 代码 利用Enumeration来适配Iterator,外部只需要调用这个适配器,即可以像调用Iterator那样,…

代码随想录算法训练营 DAY 16 | 104.二叉树最大深度 111.二叉树最小深度 222.完全二叉树的节点个数

104.二叉树最大深度 深度和高度 二叉树节点的深度:指从根节点到该节点的最长简单路径边的条数或者节点数(取决于深度从0开始还是从1开始)二叉树节点的高度:指从该节点到叶子节点的最长简单路径边的条数或者节点数(取…

模板高级使用(非类型模板参数,特化,分离编译)

文章目录 模板没有实例化取内嵌类型报错问题非类型模板参数模板的特化函数模板的特化类模板的特化1.全特化2.偏特化 模板的分离编译 模板没有实例化取内嵌类型报错问题 首先在这里分享一个模板的常见报错问题。就是模板的在没有实例化的情况下去取模板类里面的内嵌类型这时候的…

代码随想录算法训练营第十六天|104.二叉树的最大深度、559.n叉树的最大深度、111.二叉树的最小深度、222.完全二叉树的节点个数

代码随想录算法训练营第十六天|104.二叉树的最大深度、559.n叉树的最大深度、111.二叉树的最小深度、222.完全二叉树的节点个数 104.二叉树的最大深度 给定一个二叉树 root ,返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数…

【绘图案例-绘图的样式 Objective-C语言】

一、接下来呢,我们来说这个绘图的样式, 1.在这个里边呢,我们现在练习,画弧,画完了, 还有一个圆环,思路1:画一个圆,设置线宽,这个线宽,实际上,就涉及到样式的问题了,思路2:画两个不同颜色的同心圆, 也就是说,我在这个模拟器里边,我先画一个黑色的圆,然后,中间…

手机投屏到电脑

手机投屏到电脑 Github 有2个开源的手机投屏项目: Scrcpy: https://github.com/Genymobile/scrcpy QtScrcpy: https://github.com/barry-ran/QtScrcpy 这2个项目都很好用,我这里用的是 Scrcpy: 官方文档中介绍了如何在windows上使用 Scrcpy…

手机携号转网查询接口-API接口-高并发批量检测实时接口

面对携号转网业务量激增带来的海量查询需求,我们的携号转网查询接口在设计之初就立足于高性能、高可用的核心原则,力求在极端条件下的稳定性和并发处理能力达到业界领先水平。 近期,我们对携号转网查询接口进行了深度的压力测试与并发性能优…

Python:柱状-折线图

写论文,需要画数据分析图: 用柱状图描述算法执行时间用折线图描述性能改进 示例代码: import numpy as np import matplotlib.pyplot as plt from matplotlib.pyplot import MultipleLocatorSecurity ["128", "192",…

敢为天下先!深圳市全力推动鸿蒙生态发展……程序员

3月19日,鸿蒙生态创新中心揭幕仪式在深圳正式举行。鸿蒙生态创新中心的建立是为构建先进完整、自主研发的鸿蒙生态体系,将深圳打造为鸿蒙生态策源地、集聚区的具体举措,也是推动我国关键核心技术高水平自立自强、数字经济高质量发展、保障国家…

C++ —— 内存管理

目录 1. C内存分布 2. C 内存管理方式 2.1 new 和 delete 操作内置类型 2.2 new 和 delete 操作自定义类型 3. operator new与operator delete函数 4. new和delete的实现原理 5. malloc/free 和 new/delete 的区别 1. C内存分布 首先看一段代码: int globalV…

数字电源浅析

电力电子技术是关于能量转换、调节、控制和管理等方面的学科,而数字电源则是电力电子技术的一种应用,是利用数字电路技术实现电源控制和管理的新型电源。 一、什么是数字电源 数字电源是一种数字控制的电源设备,可以通过数字控制芯片(DSP、MCU等)实现输出电压、电流、功…

GPT-5揭秘:Lex Fridman与Sam Altman播客热议,AGI时代的新变革即将来临!

嘿,朋友们,你们知道吗?Lex Fridman和Sam Altman又聚在一起了,这次是在播客上。 在播客中,他们聊了很多,包括董事会的幕后故事、Elon Musk的诉讼案,甚至还提到了Ilya、Sora这些名字。 但真正让…