结构体||联合体

1.结构体

1.1实际生活中一些东西往往有多个元素组成。如一名学生有身高、体重、名字、学号等。这时候就需要用到结构体。

结构体是一些值的结合,这些值被称为成员变量。结构体的每个成员可以是不同类型的变量,如:标量、数组、指针、甚至是其他结构体。

1.2结构的声明(struct是结构体关键字)

struct tag//结构体类型

{

        member-list//结构体成员

} variable-list//变量

如描述一个学生时

1.先定义结构体类型,在定义结构体变量

struct student

{

        char name[20];

        double height;

        int age;

} ;

struct student stu1,stu2;

2.定义结构体和变量 

struct student

{

        char name[20];

        double height;

        int age;

}stu1,stu2 ;

3.直接定义变量(匿名结构体)

struct

{

        char name[20];

        double height;

        int age;

}stu1;

·匿名结构体如果没有对结构体进行重命名的话,基本上只能使用一次。 

·结构体定义不分配地址,结构体变量会分配地址。

·结构体变量的声明必须在主函数上或者主函数中

1.3结构体的初始化(基于上述描述学生代码)

int main()

{

        struct stu s1={"zhangsan",1.85,18};

        struct stu s2={"lisi",1.78,16};

        ……;

}

1.4结构体的访问

1.结构体成员的直接访问:成员访问操作符(.)

 s1.name

 s1.age

2. 结构体成员间接访问:

stuct student*p=&s1;

p->age=18;

p->name="zhangsan"

1.5typedef关键字与结构体

typedef struct student

{

        char name[20];

        int age;

} stu;//相当于struct student;

stu s1={"zhangsan",18};

stu*p=&s1;

typedef主要目的是使结构体表达更加简洁,可以理解为给结构体重命名。

 1.6结构体的自引用

struct student

{

        int age;

        struct student;

}; 

上述写法正确吗?我们从sizeof(student)角度来分析,一个结构体包含自己显然它的大小就是无穷大,这是不合理的。所以想要一个结构体进行自引用,应该让结构体包含自己的指针

struct student

{

        int age;

        struct student*NEXT;

} ;

2.1结构体内存对齐

结构体的创建与初始化我们已经掌握了,接下来我们来探讨一个问题:计算结构体的大小,这就联系到结构体内存对齐

1.对齐规则

1.1结构体的第一个成员对齐到和结构体变量起始位置偏移量为0的地址出
1.2其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处

          对齐数=编译器默认对齐数和该成员变量大小的较小值

          -VS中默认对齐数是8

          -Linux和gcc中没有默认对齐数,对齐数就是成员自身的大小

1.3结构体的总大小为最大对齐数的整数倍
1.4如果一个结构体中嵌套了别的结构体,嵌套结构体对齐到自己成员中最大对齐数的整数倍

 

由上面两张图可以发现结构体成员定义先后顺序不同,结构体的大小也不一样。这是为什么呢?接下来就用结构体内存对齐来讲解下。(VS环境)

 

第一幅图a填在偏移量为0的地方,i为int类型对齐数为4所以必须填在4的整数倍的位置,所以i填在4处,b为char类型对齐数为1所以只要填在1的整数倍的位置 所以接在i下面填在8出但是0~8的大小为9,该结构体中最大对齐数是i的对齐数4,9不是4的倍数所以结构体最后大小是12。

图二同理,可以自己去验证下。

2.为什么会存在内存对齐

1.平台原因(移植原因):

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

2.性能原因

数据结构(尤其是栈)应该是尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内容,处理器需要作两次内存访问;而对齐的内容访问仅需要访问一次。假设处理器总是从内存中去8个字节,则地址必须是8的倍数,如果我们能保证将所有的double类型的数据都对齐成8的倍数,那么就可以用一个内存操作来读或者写值了。否则,我们可能需要执行两次访问,因为对象可能被分在两个8字节内存块中。

总的来说:结构体内存对齐就是那空间来换时间的做法。

所以如果我们在设计结构体的时候既要满足对齐又要节省空间,就要对定义成员的顺序进行一定的规划。如上面两张图的差别。

3.修改默认对齐值(利用#pragma)

#pragma pack(4)//修改默认对齐数为4

struct  s

{

        ……

};

#pragma pack()//取消设置的对齐数,还原为默认

int main()

{

}

3.1结构体实现位段

1.什么是位段
1.1位段的成员必须是Int、unsigned Int、signed int 、char。在c99中位段成员的类型也可以选择其他类型 。
1.2位段的成员后边有一个冒号和一个数字。

位段的目的:节省空间

struct A

{

        int _a:2;

        int _b:5;这里的数字是指比特位(bit)

        int _c:10;

        int _d:30;

};

A就是位段的类型。位段A所占大小为8字节(原理:2+5+10+30=47bit ,一个int(整型)四个字节,32个比特位,显然一个整型不足以存放位段A,所以需要两个整型,也就是8个字节)

 1.4位段的空间上是按照需要以4个字节(int)或者1个字节(char)的方式来开辟。
1.5位段涉及很多不确定的因素,位段是不跨平台的,注重可移植的程序应该避免使用位段
1.6开辟位段空间

        VS环境下的位段开辟:1.从右向左使用。

                                              2.如果剩余的空间不够下一个成员使用就浪费。

struct S

{

        char a:3;

        char b:3;

        char c:4;

        char d:5;

};

struct S s={0};

s.a=10;

s.b=12;

s.c=6;

s.d=8;

 

可以看到内存中确实是我们所推理的那样,存放22 06 08 

2联合体

2.1联合体的声明

联合体也和结构体一样由一个或者多个元素组成,联合体的特点是所有成员共用一块内存空间。所以联合体也叫共用体 

 union Un

{

        char a;

        int i;

};

int main()

{

        union Un un={0};//联合变量定义

        printf(“%d”,sizeof(un));//4   

}

2.2联合体的特点

为什么上述声明的联合体大小是4个字节,上面讲到联合体成员是共用一块内存空间的,这样就使联合体的大小,至少是最大成员的大小。

在描述不同的商品比如:图书,杯子,衬衫时。每一种商品都有价格、库存量等但是他们又都有单独特有的属性页数、设计、尺寸。这个时候如果用结构体来描述就会十分复杂,而用联合体就可以节省空间。

最后来利用联合体特有的特性来实现上一章的大小端判断

int judge()

{

        union  un

        {

                int i;

                char a;//a占一个字节在联合体中与i的第一个字节共用一块内存空间

        }

        un.i=1;

        return un.a;

}

int main()

{

        if(judge())

                printf("小端”);

        else

                printf("大端");

}

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

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

相关文章

redis——布隆过滤器

一:布隆过滤器是什么? 由一个初值都为零的bit数组和多个哈希函数构成,用来快速判断集合中是否存在某个元素,不保存数据信息,只是在内存中做一个是否存在的标记 二:布隆过滤器能干什么? 高效…

中国湖泊面积-水位长时序数据产品(2000-2020)

今天我们分享中国湖泊面积-水位长时序数据产品(2000-2020) 该数据集包含中国典型湖泊2000-2020年最大水体面积、多年平均面积、水位变化速率及空间分布矢量。 数据溯源信息 「数据来源描述」Landsat、HJ、ZY、Jason、ENVISAT、Cryosat、ICESat和HY 「数…

dockerfile文件:copy和add 异同

相同点: 复制文件或目录: 无论是 COPY 还是 ADD 都可以将文件或目录从构建上下文复制到容器中。支持源路径和目标路径: 两者都需要指定源路径和目标路径,用于指定要复制的文件或目录在主机上的位置以及在容器中的目标路径。 不同…

RocketMQ事务消息源码解析

RocketMQ提供了事务消息的功能,采用2PC(两阶段协议)补偿机制(事务回查)的分布式事务功能,通过这种方式能达到分布式事务的最终一致。 一. 概述 半事务消息:指的是发送至broker但是还没被commit的消息,在半…

<软考>软件设计师-1计算机组成与结构(总结)

(一)计算机系统基础知识 1 计算机硬件组成 计算机的基本硬件系统由运算器、控制器、存储器、输入设备 和 输出设备 5大部件组成。 1 运算器、控制器等部件被集成在一起统称为中央处理单元(CPU) 。CPU是硬件系统的核心,用于数据的加工处理,能完成各种算…

2023.11.27【读书笔记】|医疗科技创新流程(前言)

目录 注重价值关键要素如何解决价值问题?注重三个关键点价值探索价值预测价值定位 中国视角背景挑战战术 洞察过程发现需求发现需求筛选 发明概念产生概念选择 发挥战略发展商业计划 注重价值 在美国,医疗费用的增长率已经多年超过GDP增长率&#xff1b…

Redis--11--Redis事务的理解

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 Redis事务事务回滚机制Redis 事务是不支持回滚的,不像 MySQL 的事务一样,要么都执行要么都不执行; Redis的事务原理 Redis事务 …

第九节HarmonyOS 常用基础组件1-Text

一、组件介绍 组件(Component)是界面搭建与显示的最小单位,HarmonyOS ArkUI声名式为开发者提供了丰富多样的UI组件,我们可以使用这些组件轻松的编写出更加丰富、漂亮的界面。 组件根据功能可以分为以下五大类:基础组件…

微服务实战系列之MemCache

前言 书接前文,马不停蹄,博主继续书写Cache的传奇和精彩。 Redis主要用于数据的分布式缓存,通过设置缓存集群,实现数据的快速响应,同时也解决了缓存一致性的困扰。 EhCache主要用于数据的本地缓存,因无法保…

优先队列详解

优先队列是计算机科学中的一种抽象数据类型,它是一种队列:元素拥有优先级,优先级最高的元素最先得到服务;优先级相同的元素按照在集合中的顺序得到服务。优先队列有两种主要的实现方法:堆和二叉搜索树。 简单来说&…

基于社区电商的Redis缓存架构-写多读多场景下的购物车缓存架构

社区电商的购物车缓存架构 在购物车中的功能主要有这几个:商品加入购物车、查看购物车列表、删除购物车商品、选中购物车商品进行结算 这里购物车的场景和之前用户信息以及菜谱分享信息还不同,如果在举办了大型购物活动时,购物车可能需要面…

Web(7)内网渗透

一.内网配置 网络情况 在进行IP配置的时候,根据其情况需要在不同的网卡进行配置。如果采用桥接模式的话,需要在以太网网卡上进行配置 期间有个问题,即window Server的静态IP老是分配不成功,原来是与虚拟网卡1DHCP分配…

号称要做人民货币的Spacemesh,有何新兴叙事?

​打开Spacemesh的官网,率先映入眼帘的是一个响亮的口号——On a quest to become the people’s coin(致力于成为人民的货币)!Spacemesh 联合创始人 Tomer Afek 曾表示“Spacemesh 的低准入门槛和激励兼容性,激发了从…

BootLoader升级过程讲解与串口升级案列

一、芯片选择 STM32F103RCT6 FLASH容量:512K RAM容量:48K 二、升级方式选择: 串口升级、网口升级、4G升级、SD卡升级等等。 1、SD卡升级属于升级文件事先存储在外部FLASH,不需要考虑获取升级文件的代码和升级文件存放的位置&am…

前端面试JS— JS数据类型及相关内容

目录 JS数据类型 基本数据类型: 引用数据类型: 两种数据存储方式: 两种数据类型的区别: 数据类型的检测方式 null和undefined区别 JS数据类型 基本数据类型: Number,String,Boolean,…

嵌入式 C 语言中的全局变量问题

大家好,今天分享一篇关于嵌入式C编程中全局变量问题的文章。希望对大家有所启发。 嵌入式特别是单片机os-less的程序,最易范的错误是全局变量满天飞。 这个现象在早期汇编转型过来的程序员以及初学者中常见,这帮家伙几乎把全局变量当作函数形…

十种接口安全方案!!!

其他系列文章导航 Java基础合集数据结构与算法合集 设计模式合集 多线程合集 分布式合集 ES合集 文章目录 其他系列文章导航 文章目录 前言 一、数据加密,防止报文明文传输。 二、数据加签验签 2.1 什么是加签验签呢? 2.2 有了https等加密数据&am…

Azure Machine Learning - 使用.NET创建和管理AI搜索功能

本文介绍了如何在 Azure SDK for .NET 中使用 C# 和 Azure.Search.Documents客户端库来创建和管理搜索对象。 关注TechLead,分享AI全维度知识。作者拥有10年互联网服务架构、AI产品研发经验、团队管理经验,同济本复旦硕,复旦机器人智能实验室…

【DPDK】Trace Library

概述 跟踪是一种用于了解运行中的软件系统中发生了什么的技术。用于跟踪的软件被称为跟踪器,在概念上类似于磁带记录器。记录时,放置在软件源代码中的特定检测点会生成保存在巨大磁带上的事件:跟踪文件。稍后可以在跟踪查看器中打开跟踪文件…

java学习part29线程通信

139-多线程-线程间的通信机制与生产者消费者案例_哔哩哔哩_bilibili 1.等待唤醒 类似于golang的channel, 1.1用法 类似于go的wait(), 1.sleep和wait的一个重大区别是,sleep不会让线程失去同步监视器,而wait会释放 2.wait必须tr…