结构体,枚举,联合大小的计算规则

news2025/1/11 6:04:36

目录

1.结构体大小的计算

补充(位段)

2.枚举的大小(4个字节)

3.联合大小的计算


1.结构体大小的计算

(1)结构体内存对齐的规则

1. 第一个成员在与结构体变量偏移量为 0 的地址处。


2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数 = 编译器默认的对齐数 与 该成员大小的 两者之间的较小值
VS 中默认的值为 8     gcc没有默认对齐数,那么对齐数就是该成员的大小


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


4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。

例如

struct S1
{
	char c1;//大小为1,默认对齐数为8,取其小就是1,所以其对齐数是1
	int a;//大小为4,默认对齐数为8,取其小为4,所以其对齐数就是4
	char c2;
}

int main()
{
	struct S1 s1={0};
	printf("%d\n",sizeof(s1));
}

 其内存分配如图所示

 其中的成员占了9个内存单元,但是结构体的内存大小是其最大对齐数的整数倍,这里的最大对齐数是4,所以在9以上的,最小的,4的倍数是12

再来一例,与上面同理,直接看图

struct S2
{
	char c1;
	char c2;
	int a;
}

int main()
{
	struct S2 s2={0};
	printf("%d\n",sizeof(s2));
}

 成员的内存总共为8个内存单元,刚好也是4的倍数

可以看到两个结构体的成员是相同的,但是后者占用内存比前者少,所以总结起来

将占用内存小的成员写在前面 ,这样能更加高效的利用空间

(2) 那么既然有空间的浪费,为什么不紧密的存放每一个成员呢,内存对齐的意义

1. 平台原因 ( 移植原因 ) :
不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
2. 性能原因 :
数据结构 ( 尤其是栈 ) 应该尽可能地在自然边界上对齐。
原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。
总体来说: 结构体的内存对齐是拿空间 来换取 时间 的做法。
 

 (3)修改默认对齐数

结构在对齐方式不合适的时候,可以自己更改默认对齐数。

#pragma 这个预处理指令,可以改变默认对齐数,例如

#pragma pack(1)设置默认对齐数为1,这样每一个结构体成员都是紧密排列的了

#pragma pack(1)//设置默认对齐数为1
struct S2
{
 char c1;
 int i;
 char c2;
};
#pragma pack()//取消设置的默认对齐数,还原为默认

补充(位段)

位段的内存分配
1.位段的成员可以是intunsignedint signedint或者是char(属于整形家族)类型

2.位段的空间上是按照需要以4个字节(int)或者1个字节(char)的方式来开辟的

3.位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段

struct S
{
	int a;
	int b;
	int c;
	int d;
};

按照上面的结构体的计算规则,那么这个结构体的大小为16

对于位段是8个字节

struct S
{
	int a:2;//冒号后的数字表示a只需要2个比特位
	int b:5;
	int c:10;
	int d:30;
}
//总共需要47个比特位,那么6个字节足够(6*8)=48,计算的结果却为8个字节

 注意位段是按整型大小的字节进行开辟的

注意:

struct S
{
	int a:2;
	int b:5;
	int c:10;
	int d:33;//不能超过32个比特,报错:d的位域大小无效
}

 本来int类型需要占用32位空间,现在只需要占用2个比特位的空间,位段的使用节省了空间大小

 举例

struct S
{
	char a:3;
	char b:4;
	char c:5;
	char d:4;
}

int main()
{
	struct S s={0};
	
	s.a=10;//二进制序列1010
	s.b=20;
	s.c=3;
	s.d=4;
	
	return 0;
	
}

 首先struct S s={0};所以先给s分配8个比特位

 如上图所示a只有3个比特位,但是s.a=10//二进制序列为1010,所以内存中只能存3个比特位,即1010中的010

s.b=20//二进制序列10100占5个比特,但是b只有4个比特,所以只能传“0100”

s.c=3//二进制序列011占3个比特,但是c有5个比特,前面补0,即“00011”

s.d=4//二进制序列为100,但是d占4个比特,前面补0,即“0100”

 所以总体写下来如图所示

阅读内存时,内存是16进制数字,则四个2进制位为一个16进制,从左往右依次读

0010 0010 0000 0011 0000 0100

2       2       0       3       0       4   

位段的跨平台问题

1. int 位段被当成有符号数还是无符号数是不确定的。
2. 位段中最大位的数目不能确定。
( 16 位机器最大 16 , 32 位机器最大 32 ,写成 27 ,在 16 位机器会出问题。
3. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。
4. 当一个结构包含两个位段,第二个位段成员比较大,无法容纳于第一个位段剩余的位时,是舍弃剩余的位还是利用,这是不确定的

总结

跟结构相比,位段可以达到同样的效果,可以很好的节省空间,但是有跨平台的问题存在

应用:用位段封装网络上传输的数据包等

2.枚举的大小(4个字节)

enum Color//颜色
{
 RED=1,
 GREEN=2,
 BLUE=4
};
enum Color clr = GREEN;//只能拿枚举常量给枚举变量赋值,才不会出现类型的差异。
clr = 5;
enum Sex
{
	MALE,
	FEMALE,
	SECRET
}

int main()
{
	enum Sex s=MALE;
	printf("%d\n",sizeof(s));
	
	return 0;	
} 

结果为4

3.联合大小的计算

(1)计算规则

联合的大小至少是最大成员的大小。

当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍

union Un
{
	int a;//成员大小为4,默认对齐数为8,对齐数为4
	char arr[5]; //成员大小char是1,默认对齐数是8,得到对齐数是1,不要拿数组的大小进行计算
}

int main()
{
	union Un u;
	printf("%d\n",sizeof(u));
	return 0;
}

 如果联合体如下图所示

那么其占用内存总大小为5个内存单元,但是联合的大小

至少是最大成员的大小。

当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍

5(最大成员大小)不是(最大对齐数)4的整数倍,所以应该按下图进行内存分配

如图浪费了3个内存单元,使得总内存为8个字节,为最大字节数(4)的整数倍 

 

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

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

相关文章

走出焦虑,拥抱未来!

方向一:简述自己的感受 大三科班在读,我能够理解在看到他人完成自己做不出来的题目或写不出的代码时会感到焦虑的心情,因为我也常常会有这种焦虑。这种焦虑可能是因为觉得自己与他人存在差距或者担心自己的能力不足。同时,周围人…

深度学习Redis(3):主从复制

前言 在前面的两篇文章中,分别介绍Redis内存模型和Redis持久化 在Redis的持久化中曾提到,Redis高可用的方案包括持久化、主从复制(及读写分离)、哨兵和集群。其中持久化侧重解决的是Redis数据的单机备份问题(从内存到…

Dockerfile构建Tomcat镜像(源码)

Dockerfile构建Tomcat镜像 目录 Dockerfile构建Tomcat镜像 1、建立工作目录 2、编写Dockerfile文件 3、构建镜像 4、测试容器 5、浏览器访问测试: 1、建立工作目录 [roothuyang1 ~]# mkdir tomcat[roothuyang1 ~]# cd tomcat/[roothuyang1 tomcat]# lsapach…

GLM模型介绍

paper: 《GLM: General Language Model Pretraining with Autoregressive Blank Infilling》 摘要: 我们提出了一个基于自回归空白填充的通用语言模型(GLM)来解决这一挑战。GLM通过添加2D位置编码和允许任意顺序预测跨度来改进空白填充预训…

静态网页加速器:优化性能和交付速度的 Node.js 最佳实践

如何使用 Node.js 发布静态网页 在本文中,我们将介绍如何使用 Node.js 来发布静态网页。我们将创建一个简单的 Node.js 服务器,将 HTML 文件作为响应发送给客户端。这是一个简单而灵活的方法,适用于本地开发和轻量级应用。 1、创建静态网页…

答辩PPT怎么做?在线PPT软件哪个好?

又是一年毕业季,相信很多毕业生都开始准备论文答辩,有些同学正在为论文奋夜苦战,有些则是为论文答辩PPT而烦恼。做PPT要用什么软件好呢?这篇文章就来告诉你。 当下有很多PPT制作工具,其中自然也包括Office三件套。这些…

手机卡线上销户难,高额违规金,一招教运营商做人!

今天有感而发,运营商大哥们,请温柔点对待你的客户好吗?少一点的套路,多一点的人性化,这不比什么都强吗?客户想要销户了,千方百计的阻拦,这样不要吧,没多大意义啊&#xf…

达芬奇架构 DaVinci Core - 小记

文章目录 官方文档 : HUAWEI Da Vinci Architecture https://support.huaweicloud.com/intl/en-us/odevg-A800_9000_9010/atlaste_10_0007.htmlPPT : DaVinci: A Scalable Architecture for Neural Network Computing https://www.cmc.ca/wp-content/uploads/2020/0…

《Ansible自动化工具篇:Centos操作系统基于ansible工具一键远程离线部署之K8S1.24.12二进制版集群》

一、部署背景 由于业务系统的特殊性,我们需要针对不同的客户环境部署二进制版K8S集群,由于大都数用户都是专网环境,无法使用外网,为了更便捷,高效的部署,针对业务系统的特性,我这边编写了 基于a…

【OJ比赛日历】快周末了,不来一场比赛吗? #08.05-08.11 #15场

CompHub[1] 实时聚合多平台的数据类(Kaggle、天池…)和OJ类(Leetcode、牛客…)比赛。本账号会推送最新的比赛消息,欢迎关注! 以下信息仅供参考,以比赛官网为准 目录 2023-08-05(周六) #7场比赛2023-08-06…

深度学习论文分享(六)Simple Baselines for Image Restoration

深度学习论文分享(六)Simple Baselines for Image Restoration 前言Abstract1 Introduction2 Related Works2.1 Image Restoration2.2 Gated Linear Units 3 Build A Simple Baseline3.1 Architecture3.2 A Plain Block3.3 Normalization3.4 Activation3…

新手指南:流程图中各种图形的含义及用法解析

我们经常在技术设计、沟通、业务演示等一些领域看到流程图,它也可以称为输入输出图。顾名思义,它是指一种简单的工作流程的具体步骤,比如包括一次会议的流程,以及一次生产制造的顺序和过程等。本文将为大家介绍流程图的含义和具体…

通向架构师的道路之tomcat集群

一、为何要集群 单台App Server再强劲,也有其瓶劲,先来看一下下面这个真实的场景。 当时这个工程是这样的,tomcat这一段被称为web zone,里面用springws,还装了一个jboss的规则引擎Guvnor5.x,全部是ws没有se…

python怎么保存代码文件,python中怎么保存代码

大家好,给大家分享一下怎么保存python代码写好怎么运行,很多人还不知道这一点。下面详细解释一下。现在让我们来看看! 广告关闭 腾讯云11.11云上盛惠 ,精选热门产品助力上云,云服务器首年88元起,买的越多返…

c++11 标准模板(STL)(std::basic_ofstream)(二)

定义于头文件 <fstream> template< class CharT, class Traits std::char_traits<CharT> > class basic_ifstream : public std::basic_istream<CharT, Traits> 类模板 basic_ifstream 实现文件流上的高层输入操作。它将 std::basic_istrea…

windows环境下安装RabbitMQ

一、RabbitMq简介1.1消息队列中间件简介消息队列中间件是分布式系统中重要的组件&#xff0c;主要解决应用耦合&#xff0c;异步消息&#xff0c;流量削锋等问题实现高性能&#xff0c;高可用&#xff0c;可伸缩和最终一致性[架构] 使用较多的消息队列有 ActiveMQ(安全)&#x…

Java并发编程之顺序一致性

如果程序是正确同步的&#xff0c;程序的执行将具有顺序一致性&#xff08;Sequentially Consistent&#xff09;——即程序的执行结果与该程序在顺序一致性内存模型中的执行结果相同。 同步&#xff0c;即排队。 同一时刻&#xff0c;只能有一个线程和内存交互&#xff01;&a…

基于C#的窗体阴影效果方案 - 开源研究系列文章

最近在研究C#的Winform窗体的效果&#xff0c;上次介绍了窗体动画效果的博文( 基于C#的无边框窗体动画效果的完美解决方案 - 开源研究系列文章 )&#xff0c;这次将窗体阴影效果的方案进行一个介绍。 找了一下度娘&#xff0c;具体窗体阴影效果就两种方法&#xff1a;直接绘制和…

机器学习深度学习——池化层

&#x1f468;‍&#x1f393;作者简介&#xff1a;一位即将上大四&#xff0c;正专攻机器学习的保研er &#x1f30c;上期文章&#xff1a;机器学习&&深度学习——卷积的多输入多输出通道 &#x1f4da;订阅专栏&#xff1a;机器学习&&深度学习 希望文章对你们…

powershell 自动补全

我在zsh中使用智能提示 zsh-autosuggestions已经习惯了&#xff0c;而在windows中的终端默认没有智能提示&#xff0c;很起来不舒服。那么有没有什么替代呢&#xff1f; 安装powershell-7.3 我的windows 10默认安装的是powershell-5.1.19&#xff0c;所以我们先来升级一下。 …