【C语言】自定义类型:结构体深入解析(二)结构体内存对齐宏offsetof计算偏移量结构体传参
文章目录
- 📝前言
- 🌠 结构体内存对齐
- 🌉内存对齐包含结构体的计算
- 🌠宏offsetof计算偏移量
- 🌉为什么存在内存对⻬?
- 🌠 结构体传参
- 🚩总结
📝前言
本小节,我们学习结构的内存对齐,理解其对齐规则,内存对齐包含结构体的计算,使用宏offsetof
计算偏移量,为什么要存在内存对齐?最后了解结构体的传参文章干货满满!学习起来吧😃!
🌠 结构体内存对齐
结构体内存对齐指的是结构体中各成员变量在内存中的存储位置按照一定规则对齐。
既然是按照一定规则,那得首先了解它的对齐规则:
- 结构体的第一个成员对齐到和结构体起始位置偏移量为
0
的地址处。 - 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数 = 编译器默认的一个对齐数 与 该成员变量大小的较小值。
- VS 中默认的值为 8
linux
中gcc
没有默认对齐数,对齐数就是成员自身的大小
- 结构体总大小为最大对齐数(结构体中的每一个成员都有一个对齐数,所有对齐数中的)的整数倍。
- 如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。
- 来代码理解:
struct S1
{char c1;char c2;int i;
};struct S2
{char c1;int i;char c2;
};int main()
{printf("%d\n", sizeof(struct S1));printf("%d\n", sizeof(struct S2));return 0;
}
代码运行:
分析:
首先结构体S1
的成员有三个,根据对齐规则:结构体的第一个成员对齐到结构体变量起始位置偏移量为0
的地址处-—>C1
放在偏移量为0
的地址处,接下来第二个C2
就从第2
个规则按对齐数进行放置,C2
的字节数char
类型,大小为1
,VS的默认对齐数为8,对齐数取的是默认对齐数和成员变量字节大小的较小值,1<8
,取1
为对齐数,然后偏移量为1
的位置放1
,此时再看第三个变量i的字节大小为4
,4<8
,对齐数为4,当放在偏移量为2时,2不是4的整数倍,跳过,3
也不是,跳过,而当偏移量为4时刚好是4的整数1
倍(4*1=4
),然后占据为4
个字节空间,从偏移量0
到最后偏移量的空间就是结构体的总大小,为8
,此时还没有结束,要验证,根据第三条规则结构体的总大小为最大对齐数的整数倍,最大对齐数为4
(4>1>1
),而结构刚才计算出来是8
刚好是4
整数倍(4*2
)当这些都符合了,结构体的大小就是8
了。
一个例子你可能想是不是碰巧,那么第二个例子:
结构体S2
中有三个成员,C1
大小为一,第一个成员放在偏移量为0
处,第二个成员i
大小为4
,偏移量1
,2
,3
都不是4
的整数倍,然后这些空间都跳过不放数据,(注:他开辟了空间,但他此时不用,你可能会想:这不浪费吗?文章我们慢慢解释)然后偏移量为4
时为整数倍,从偏移量4
开始放i
直到7
,第三个元素C2
大小为1
,1
的整数倍任何数的整数倍,可以直接放,当放在偏移量8
处时,全部成员都放完了,我们还要对他进行验证是否为整数倍。S2
最大对齐数是4
,偏移量9
,10
都不对,当偏移量为11
,从0
到11
刚好为12
,为4
的倍数(4*3=12
)。所以S2
总大小为12
!
🌉内存对齐包含结构体的计算
struct S3
{double d;char c;int i;
};
int main()
{printf("%zd\n", sizeof(struct S3));return 0;
}
运行结果:16
分析:
首先第一个成员为d
,放在偏移量为0
处,double
类型,大小为8
,位置范围为0 ~ 7
,第二个成员C
,类型为char
,大小为1
,1<8
,对齐数为1
,1
可以直接放,占据8
位置处,第三个成员i
,大小为4
,4<8
,对齐数是4
,偏移量9
,10
,11
都不是4
的倍数,12
开始占据4
个空间到15
,范围0 ~ 15
总大小为16。
S3结构体是三个成员(8>4>1
)大小最大是double
大小为8
,此时总结构体大小16
刚好为8
的2
倍,符合条件。
- [] 包含
S3
的结构体
struct S4
{char c1;struct S3 s3;double d;
};int main()
{printf("%zd\n", sizeof(struct S4));return 0;
}
运行结果:32
第一个成员C1
对应到偏移量为0
处,大小为1
,s3
为结构体,s3的大小为16,根据第四条规则【如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。】,也就是说结构体s3最大对齐数为double的8,用8对齐到S4中整数倍,1,2,3,4,5,6,7都不是8的整数倍,跳过,当偏移量为8时为对齐数8的整数倍时,然后结构体整体大小为16,占据范围为8 ~ 23,接下来就是第三个元素d
,大小为8
,偏移量24
就是8
的整数倍,占据了24 ~ 31
,所有成员都完成了,偏移量范围在0 ~ 31
,总大小就是32
。答案就是32
.看到这里的你,给自己鼓个掌,继续加油。
🌠宏offsetof计算偏移量
宏offsetof可以用来计算结构体成员相对于结构体起始位置的偏移量。
宏offsetof原型:
offsetof(type, member)
type是结构体类型
member是结构体中的成员。
注意:使用offsetof
宏计算结构体成员偏移量时,需要包含stddef.h
头文件
# define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <stddef.h>
struct S1
{char c1;char c2;int i;
};struct S2
{char c1;int i;char c2;
};struct S3
{double d;char c;int i;
};struct S4
{char c1;struct S3 s3;double d;
};int main()
{struct S1 s1 = {0};//8struct S2 s2 = { 0 };//12printf("结构体大小:\n");printf("S1=%zd\n", sizeof(struct S1));//8printf("S2=%zd\n", sizeof(struct S2));//12printf("S3=%zd\n", sizeof(struct S3));//16printf("S4=%zd\n", sizeof(struct S4));//32printf("\n"); printf("结构体S1成员的偏移量:\n");printf("c1=%zd\n", offsetof(struct S1, c1));//0printf("c2=%zd\n", offsetof(struct S1, c2));//1printf(" i=%zd\n", offsetof(struct S1, i));//8printf("\n");printf("结构体S2成员的偏移量:\n");printf("c1=%zd\n", offsetof(struct S2, c1));//0printf(" i=%zd\n", offsetof(struct S2, i));//4printf("c2=%zd\n", offsetof(struct S2, c2));//8printf("\n");printf("结构体S4成员的偏移量:\n");printf("c1=%zd\n", offsetof(struct S4, c1));//0printf("s3=%zd\n", offsetof(struct S4, s3));//8printf("d=%zd\n", offsetof(struct S4, d));//24return 0;
}
运行+图对比:
🌉为什么存在内存对⻬?
- 平台原因 (移植原因):
不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。 - 性能原因:
数据结构(尤其是栈)应该尽可能地在⾃然边界上对⻬。原因在于,为了访问未对⻬的内存,处理器需要作两次内存访问;⽽对⻬的内存访问仅需要⼀次访问。
假设⼀个处理器总是从内存中取
8
个字节,则地址必须是8
的倍数。如果数据没有对齐,CPU需要额外的时间来处理非对齐的内存访问,这会降低性能。
总结一句话来说:
结构体的内存对⻬是拿空间来换取时间的做法。
在设计结构体时,既要满足内存对齐要求,又要考虑节省空间,可以采取以下方法:
- 尽量将较小类型如
char
、short
等成员放在结构体开始位置。这可以减少由对齐产生的内存浪费。
例如前面的S1
和S2
就很典型:
struct S1
{char c1;int i;char c2;
};
struct S2
{char c1;char c2;int i;
};
阿森把宝图解:
- 修改默认对⻬数
#pragma
这个预处理指令,可以改变编译器的默认对⻬数。
#pragma
原型:
#pragma pack(push, 1) // 将结构体对齐数设置为1字节
struct S1
{char a; int b;
};
#pragma pack(pop)// 恢复之前的对齐数
pack(push, 1)
表示将当前对齐数压入栈,并设置新的对齐数为1
字节pack(pop)
表示从栈中弹出之前的对齐数,恢复默认对齐数
可以直接指定对齐数:
#pragma pack(1)
struct S1
{ // 成员对齐数为1字节char a; int b;
};#pragma pack() // 恢复默认对齐数
例子:
#pragma pack(1)
struct S1
{char c1;char c2;int i;
};
#pragma pack()int main()
{printf("%d\n", sizeof(struct S1));return 0;
}
输出:
图解对比:
🌠 结构体传参
- 按值传递(传结构体)
函数形参声明为结构体,实参传递结构体变量。此时在函数内对形参的修改不会影响实参。
struct St
{int x;
};void func(struct St st)
{st.x = 10;
}int main()
{struct St s = { 0 };func(s);//传结构体printf("%d\n", s.x);
}
输出:
- 按地址传递
函数形参定义为结构体指针,实参传递结构体变量的地址。函数内对形参所指结构体的修改会影响实参。
struct St
{int x;
};void func(struct St* p)
{p->x = 10;
}int main() {struct St s = { 0 };func(&s);printf("%d\n", s.x);
}
输出:
- 传结构体指针
实参直接传结构体指针:
struct St
{int x;
};void func(struct St* st)
{st->x = 10;
}int main()
{struct St s;struct St* p = &s;func(p);printf("%d\n", s.x);
}
输出:10
分析:
传值也就是把整个结构体传过去,我们知道形参是是实参的一份临时拷贝,需要再创建特别大的空间来存储结构体。
无论是传结构体指针还是传结构体地址,本质上都是传地址,但是传地址,只需要创建一个小的空间来存储地址。
选择传地址比较好一些。
原因:
函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。
如果传递⼀个结构体对象的时候,结构体过⼤,参数压栈的的系统开销⽐较⼤,所以会导致性能的下降。
总结:
结构体传参的时候,要传结构体的地址。
🚩总结
这次阿森和你一起学习结构体的 结构体内存对齐,内存对齐包含结构体的计算,使用宏offsetof
计算偏移量,为什么存在内存对⻬? 结构体传参的本质,阿森将下一节和你一起学习结构体实现位段。
感谢你的收看,如果文章有错误,可以指出,我不胜感激,让我们一起学习交流,如果文章可以给你一个小小帮助,可以给博主点一个小小的赞😘
相关文章:
【C语言】自定义类型:结构体深入解析(二)结构体内存对齐宏offsetof计算偏移量结构体传参
文章目录 📝前言🌠 结构体内存对齐🌉内存对齐包含结构体的计算🌠宏offsetof计算偏移量🌉为什么存在内存对⻬?🌠 结构体传参🚩总结 📝前言 本小节,我们学习结构的内存对…...
活动回顾 (上) | 2023 Meet TVM 系列活动完美收官
作者:xixi 编辑:三羊、李宝珠 2023 Meet TVM 年终聚会于 12 月 16 日在上海圆满落幕,本次 meetup 不仅邀请到了 4 位 AI 编译器专家为大家带来了精彩的分享,还新增了圆桌讨论环节,以更多元的视角和各位共同讨论大模型…...
JMeter常见配置及常见问题修改
一、设置JMeter默认打开字体 1、进入安装目录:apache-jmeter-x.x.x\bin\ 2、找到 jmeter.properties,打开。 3、搜索“ languageen ”,前面带有“#”号.。 4、去除“#”号,并修改为:languagezh_CN 或 直接新增一行&…...
描述一个bug及定义bug的级别
(一)描述一个bug 描述一个bug,需要以下几个因素: 故障标题、故障发现的版本、故障类别(功能/兼容/界面)、故障优先级、故障描述(测试环境、测试步骤、预期结果、实际结果)。 举个例…...
Java项目-瑞吉外卖项目优化Day3
前后端分离开发 Yapi 是一个接口结合了接口测试、接口管理的管理平台,需要配置比较麻烦。看弹幕说用apifox更好用。可以将接口文档导出导入。 Swagger 注意下面的地址前面要有/。 效果: 可以在这里实现接口的测试,也可以导出文档等等。一般…...
测试理论知识四:大型软件的测试技巧——单元测试
1. 模块测试/单元测试 模块测试也被称为单元测试,本文章称单元测试为主。 对于小的程序测试,我们可以在一定时间内完成,如果面对的是大型程序,等程序开发完成之后我们再进行测试,那会大大降低我们的效率。 单元测试…...
安防监控系统/磁盘阵列/视频监控EasyCVR平台微信推送步骤大公开
视频汇聚/视频云存储/集中存储/视频监控管理平台EasyCVR能在复杂的网络环境中,将分散的各类视频资源进行统一汇聚、整合、集中管理,实现视频资源的鉴权管理、按需调阅、全网分发、云存储、智能分析等,视频智能分析平台EasyCVR融合性强、开放度…...
算法与数据结构--特殊有序集的线性时间排序算法
一.计数排序算法 基本思想:统计每个输入元素的个数,然后根据这些计数值重构原数组。 使用范围:需要知道元素大小范围,就是最大值是多少。 【排序算法】计数排序_哔哩哔哩_bilibili 二.基数排序 使用场景:只适用于…...
windows 动态库和静态库 介绍
在Windows平台上,动态库和静态库都是用于组织和共享代码的方式。这些库文件的扩展名和用途有一些区别。 1. 静态库和动态库 静态库(Static Library): 文件扩展名:.lib在编译链接时,静态库的代码被直接嵌入…...
微软官方镜像下载大全(windows iso 官方镜像)
原本只是想下一个Windows Server 2022中文版的镜像,后面发现要么就是慢得一批的某盘,要么就是磁力,我想直接下载简简单单,找了一圈没有找到。官网下载需要注册、登录乱七八糟,最终终于找到下载方法了,适用于…...
ceph块存储学习
目录 ceph的组件和功能 ceph的数据读写流程 ceph存储池学习 ceph的组件和功能 Ceph OSD:功能是存储数据,处理数据的复制、恢复、平衡数据分布,并将一些相关数据提供给Ceph Monitor,。 Ceph Monitor: 功能是维护整个集群健康状态&…...
开发模型和测试模型
1. 开发模型 1.1 瀑布模型 瀑布模型是其他模型的基础框架 start—>需求分析---->计划----->设计----->编码----->测试----->End(其实就是软件开发的生命周期) 特点:线性的开发流程 缺陷:测试被后置。①风险往…...
Kubectl 部署简单应用
创建新服务 kubectl create deployment kubernetes-bootcamp --imagegcr.io/google-samples/kubernetes-bootcamp:v1 查看 kubectl get deployments 打开新的终端执行 kubectl proxy 此时,切回上一个终端,通过 kubectl get pods 可查看已部署好的pod。并…...
Flink电商实时数仓(三)
DIM层代码流程图 维度层的重点和难点在于实时电商数仓需要的维度信息一般是动态的变化的,并且由于实时数仓一般需要一直运行,无法使用常规的配置文件重启加载方式来修改需要读取的ODS层数据,因此需要通过Flink-cdc实时监控MySql中的维度数据…...
四种消息队列,如何选型
这篇文章,主要讲述 Kafka、RabbitMQ、RocketMQ 和 ActiveMQ 这 4 种消息队列的异同,无论是面试,还是用于技术选型,都有非常强的参考价值。 01 消息队列基础 1.1 什么是消息队列? 消息队列是在消息的传输过程中保存消…...
flutter开发windows应用的库
一、window_manager 这个插件允许 Flutter 桌面应用调整窗口的大小和位置 地址:https://github.com/leanflutter/window_manager二、win32 一个包,它使用FFI包装了一些最常见的Win32 API调用,使Dart代码可以访问这些调用,而不需…...
机器学习--线性回归
目录 监督学习算法 线性回归 损失函数 梯度下降 目标函数 更新参数 批量梯度下降 随机梯度下降 小批量梯度下降法 数据预处理 特征标准化 正弦函数特征 多项式特征的函数 数据预处理步骤 线性回归代码实现 初始化步骤 实现梯度下降优化模块 损失与预测模块 …...
【Spring Boot】面试题汇总,带答案的那种
继上次的文章【MySQL连环炮,你抗的住嘛?】爆火之后,越来越多的小伙伴后台留言,要求阿Q总结下其他的“连环炮”知识点,想在金九银十的面试黄金期轻松对线面试官。 同样为了节省大家的时间,阿Q最近对【Sprin…...
【大模型】快速体验百度智能云千帆AppBuilder搭建知识库与小助手
文章目录 前言千帆AppBuilder什么是千帆AppBuilderAppBuilder能做什么 体验千帆AppBuilderJava知识库高考作文小助手 总结 前言 前天,在【百度智能云智算大会】上,百度智能云千帆AppBuilder正式开放服务。这是一个AI原生应用开发工作台,可以…...
字符串压缩
...
MsSQL中的索引到底长啥样,查找过程怎么进行
参考文章一 参考文章二 建表 mysql> create table user(-> id int(10) auto_increment,-> name varchar(30),-> age tinyint(4),-> primary key (id),-> index idx_age (age)-> )engineinnodb charsetutf8mb4;insert into user(name,age) values(张三,…...
WPF 全局异常处理
在Application中存在三种异常事件EventHandler DispatcherUnhandledExceptionAppDomain.CurrentDomain.UnhandledExceptionTaskScheduler.UnobservedTaskException 其中 DispatcherUnhandledException 是在异常由应用程序引发但未进行处理时发生,但无法捕获多线程…...
Flink系列之:Elasticsearch SQL 连接器
Flink系列之:Elasticsearch SQL 连接器 一、Elasticsearch SQL 连接器二、创建 Elasticsearch表三、连接器参数四、Key 处理五、动态索引六、数据类型映射 一、Elasticsearch SQL 连接器 Sink: BatchSink: Streaming Append & Upsert ModeElasticsearch 连接器…...
java中将Map集合、对象、字符串转换为JSON对象
1、Map集合转JSON对象 创建一个Map集合; 新建json对象,并将Map引入json中。 public void demo1(){ //创建一个Map集合Map<String, String> map new HashMap<>();map.put("1729210001","zhangsan");map.put("17292…...
理解Spring中bean的作用域
singleton:Spring Ioc容器中只会存在一个共享的Bean实例,无论有多少个Bean引用它,始终指向同一个对象,作用域为Spring中的缺省(同一package)作用域 prototype:每次通过Spring容器获取prototype定义的bean时,…...
edge中以右键“打印”的方式“保存”当前页面的pdf形式,下载过程中卡进度的问题
目录 问题描述: 可能的问题: 解决: 问题描述: 特殊情况下需要保存网页的pdf形式,但页面没有类似“导出pdf”的功能按钮,可以通过页面右键“打印”的方式“保存”当前页面的pdf形式。在pdf文件下载过程中出…...
c# 使用OpenCV
C#和OpenCV的结合主要通过一个名为OpenCVSharp的库实现。OpenCVSharp是一个C#包装器,它提供了对OpenCV(一个开源的计算机视觉和机器学习库)功能的访问。 安装OpenCVSharp NuGet包: 在Visual Studio中,右键点击你的项目…...
数据库连接问题 - ChatGPT对自身的定位
1.一段关于数据库连接的技术性对话 sweetie,连接数据库的时候,需要在每次读写数据后就把连接释放吗? 亲爱的,连接数据库后,通常会在每次读写数据后将连接释放。这是为了确保数据库连接的及时释放和有效管理。如果不及…...
常见可视化大屏编辑器有哪些?
前言: 在当今数字化时代,可视化大屏编辑器成为了数据展示和决策支持的重要工具。大屏编辑器不仅仅是数据的呈现,更是数据背后的故事的讲述者。它通过图表、图形和实时数据的呈现,为用户提供了全面的信息视图,帮助用户更…...
利用ffmpeg cv2取h265码流视频(转换图片灰屏问题解决)
利用海康威视相机拍出来的视频是H265格式的,相比于常规的H264编码,压缩率更高,但因此如果直接用正常取流方法读取,会出现无法读取的情况 1. 如图h265码流取出图片为灰屏 2 、解决灰屏问题 import subprocess import cv2# 将h265流…...
中山专业做网站公司/推广网页怎么做的
参考文献: Tomcat之the jre_home environment variable is not defined correctly this environment variable is need 第一种: 双击tomcat的bin目录下的startup.bat时一闪而过时, 可以右键startup.bat找一个文本编辑器打开,然…...
python做动态网站/加盟培训机构
引用 C中有一个很方便的语法叫做引用,作用就是使得函数能够对传入的参数作出全局有效的改动。用法很简单,就是在传入参数的类型后面加上&就可以指明传入的参数是引用。 例子: #include <stdio.h>void change(int& x){x 1; } i…...
网站开发支付功能怎么做/百度网盘客户端
我是一名影像科医生,经常需要在家或出差时浏览医院患者影像资料及书写报告,由于医院没有安装影像PACS云储存,造成离开医院就无法远程办公,为了解决这个问题,本着花小钱办大事的原则,经过多方咨询相关专业人…...
网站建设分工的通知/免费注册二级域名的网站
亲爱的小伙伴们咱们5月开课计划已出座位有限感兴趣的小伙伴赶紧预约啦建策科技5月开班计划◼ 主从复制:是用来建立一个和主数据库完全一样的数据库环境,称为从数据库;主数据库一般是准实时的业务数据库。◼ 主从复制的作用:做数据…...
合肥手机网站制作/北京seo优化厂家
一、阅读学习《Thinking in Java》这本书二、敢看学习尚学堂-高琪300集java基础视频三、学习JavaEE基础,掌握Spring框架...
零售户电商网站订货网址/美业推广平台
它发生在我们所有人身上 —— 生活变得忙碌,我们与朋友失去联系。偶尔的电话和短信只够勉强了解他们的生活和家庭。 我们求助于通过社交媒体来跟随他们,在这里我们看到他们渡过美好假期、搬到新房子或换了工作。即使您无法参加聚会,你也能看到…...