自定义类型详解 ----结构体,位段,枚举,联合
目录
结构体
1.不完全声明
2.结构体的自引用
3.定义与初始化
4.结构体内存对齐与结构体类型的大小
结构体嵌套问题
位段
1.什么是位段?
2.位段的内存分配
枚举
1.枚举类型的定义
2.枚举的优点
联合(共同体)
1.联合体类型的声明以及变量定义
2.联合体的特点
利用联合体判断当前机器是大端还是小端
3.联合体大小的计算
结构体
结构体的基础知识以前的博客已经介绍过了,这里是进阶版
1.不完全声明
在创建结构体类型的时候可以省略标签,如图
这种结构体被称为匿名结构体,如果这样创建,那我们就只能在创建类型的同时创建变量,无法在其他地方创建变量,比如这里的x就是在创建此匿名结构体类型的同时创建的变量。
如果我们再创建一次匿名结构体类型
他与上面的匿名结构体成员变量均相同,但是我在这里创建了一个结构体指针p,那么p=&x这种写法正确吗?答案是不正确,编译器会把上面的两个声明当做不同的类型。
2.结构体的自引用
有这样一段代码
对于这样一段代码,如果用sizeof来计算这种类型的大小,直接就计算不出来,因为在结构体类型里面又包含了一个同样类型的变量,那这个变量势必又会包含一个int类型的data和又一个结构体类型的变量,无限套娃,都无法计算这种类型的大小,显然这样的自引用是错误的。
正确的自引用方式是使用指针
来看下面一段代码
我们先把一个匿名结构体类型进行重命名为Node,然后使用他在主函数里面创建变量,可以吗?
答案是不可以,这就像我们遇到的经典问题,先有鸡还是先有蛋,再创建类型的时候我都还没有重命名为Node,居然就先使用了,显然是错误的。
正确写法
3.定义与初始化
可以先创建类型再初始化,也可以在创建类型的同时初始化
如图是在创建类型的同时初始化
初始化的时候可以直接像sn1一样按顺序写,这样编译器认为是按照结构体成员出现的顺序初始化的,也可以使用点操作符,这样就可以按照自己的顺序来初始化。
4.结构体内存对齐与结构体类型的大小
运行这样一段代码
我们想要计算这两个结构体类型的大小,而这个结构体类型里面成员变量有一个int类型和两个char类型,大小加起来应该是六个字节,但是我们打印出来发现,结果居然是12和8,不仅不是6,甚至都两次结果都不一样大,这是为什么呢?
通过offsetof计算出struct s1各个成员变量离起始地址的偏移量分别是0,4,8,(offsetof是一个宏,具体用法这里就不展开讲了)于是我们可以画出struct s1的成员变量在内存中的分布
那也就是占了九个字节,为什么struct s1的大小会是12呢?
这就涉及到了内存对齐
首先得掌握结构体的对齐规则:
1. 第一个成员在与结构体变量偏移量为0的地址处。
2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数 = 编译器默认的一个对齐数与该成员大小的较小值。VS中默认的值为8,gcc环境没有对齐数,如果成员变量是一个数组,则他的对齐数是数组中元素的字节大小与默认对齐数的较小值,比如char ch[5]={0};他的对齐数大小就是1
3. 结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。
4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。
知道了这些规则之后再来看上面的struct s1在内存中为什么这样存储,struct s1的第一个成员变量是char类型的,因此对齐数是1,因此第一个成员变量就放在了偏移量为零的单元,第二个成员变量i是int类型的,对齐数是4,应该放偏移量是在4的整数倍的单元处,因此从4开始放四个字节,浪费了三个字节单元,第三个成员变量也是char类型的,对齐数是1,因此接着放一个字节,如图所示
一共占用了9个字节,但是根据第三条规则,结构体类型的大小必须是所有成员变量中最大对齐数的整数倍,struct s1的成员变量最大对齐数是4,9不是4的倍数,此时离着4的倍数最近的字节数是12,因此struct s1的大小是12个字节。
同样的,再来看struct s2
他的第一个成员变量是int类型的,从偏移量为零的地址处开始存放,放了四个字节,第二个成员变量是char类型的,对齐数是1,直接接着放就行,因为任何数都是1的整数倍。第三个成员变量也是char类型的,对齐数是1,接着放一个字节,发现一共用了6个字节,最大对齐数是4,6不是4的倍数,离着最近的一个4的倍数是8,因此struct s2的大小是8
结构体嵌套问题
struct s4类型的第一个成员变量是char类型的,对齐数是1,直接在偏移量为0的地址处开始存放,第二个成员变量是struct s3类型的,他的成员变量最大对齐数是8,因此他要对齐到偏移量为8的整数倍的位置,离着最近的就是从偏移量为8的位置开始存放,存放多少个字节,那就要看struct s3类型占多少个字节。
我们假设又有一块空间存放着struct s3
struct s3的第一个成员变量是double类型,对齐数是8,从偏移量为0的地址处开始存放,存放了8个字节,存到了偏移量为7的位置,第二个成员变量是char类型的,对齐数是1,接着存一个字节,存到了偏移量为8的位置,第三个成员变量是int类型的,对齐数是4,但是接下来要存的单元偏移量是9,不是4的整数倍,因此要从偏移量为12的位置开始存放四个字节,到了偏移量为15的位置。又因为struct s3的成员变量最大对齐数是8,因此struct s3所占的字节大小就是离15最近的8的整数倍也就是16.
再回到struct s4的存储,从偏移量为8的单元开始放16个字节,到了偏移量为23的位置,struct s4的第三个成员变量是double类型的,对齐数是8,而接下来要存放的位置偏移量是24,恰好是8的整数倍,因此接着存放8个字节单元到了偏移量为31的单元处。共计占用了32个字节。所有成员变量的最大对齐数是8,而32恰好是8的整数倍,因此struct s4的大小是32个字节
为什么存在内存对齐?
1. 平台原因(移植原因):不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
2. 性能原因:数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。
总体来说结构体的内存对齐是拿空间来换取时间的做法。
注:可以通过#pragma pack(x)来设置默认对齐数为x,如果想还原默认的对齐数就再调用一次#pragma pack(),括号内什么也不写就行了
5.结构体传参
这在初阶的时候已经介绍过,可以传一个结构体变量,这样访问的时候就用点操作符,也可以传一个结构体指针,这样访问的时候就用箭头操作符,推荐使用传结构体指针的形式,因为传结构体变量会导致形参在内存中开辟一块与原来结构体一样大的内存,造成内存的浪费。再者如果是传的结构体变量,实际上他是实参的一份临时拷贝,对他进行的任何操作都不会改变实参,因而我们就无法通过调用函数来操作实参了。
位段
1.什么是位段?
位段的声明和结构是类似的,有两个不同:
1.位段的成员必须是 int、unsigned int 或signed int或char 。
2.位段的成员名后边有一个冒号和一个数字。冒号后面的数字代表这个成员变量占的比特位。注意是比特位,不是字节。
但是位段与结构体是两个不同的概念,比如
那么struct A就是一个位段类型
他表示_a占2个比特位,_b占5个比特位,以此类推。通过sizeof计算A的大小,发现是8个字节。
为什么要有位段呢?假如我只需要_a表示四种情况,那么两个比特位就足够了,但是int类型的_a我们却为他开辟了32个bit位,实际有30个bit浪费了,因此位段是一种节省空间的方法。
2.位段的内存分配
1. 位段的成员可以是 int unsigned int signed int 或者是 char (属于整形家族)类型
2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。
3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。
为什么说位段具有不确定因素呢?还是以前面的例子为代表
再创建_a的时候,开辟了4个字节也就是32个比特位,然后用了两个,剩下了30个比特位,那这里就有不确定因素
比如这2个比特位是在刚才开辟的空间的左边还是右边?
接着创建了_b变量,占5个字节,接着在剩下的30个bit里面放5个,接着创建_c又放了10个,这时候还剩下15个,此时创建_d要占去30个bit,剩下的位置已经不够放了,只能再申请一块32bit的内存,那这里又有不确定因素了,新申请的空间能直接放下_d,那么原来剩下的15个bit还放不放?是先放15个到上一块空间中,再放15个bit到新申请的空间中,还是直接放30个bit到新申请的空间中?这些都是不确定因素,C语言并没有规定。
有这样一个位段,在我使用sizeof计算这个位段大小的时候发现是3个字节,那么S在内存中就有可能是这样存的
我画的是从右往左存的,当然也可能是从左往右存的,但是目前能确定的是这里如果原来开辟的内存不够用了,再申请一块新空间的时候,原来剩下不够的那些内存没有使用
再来测试一下是从左往右还是从右往左
对s进行如图的初始化,假如是从右往左放的话,内存中的存储应该是这样
换算成16进制应该是620304,通过的调试,发现确实如此。
因此在当前环境下位段就是从右往左,不够的空间就不用的方式分配内存。
枚举
1.枚举类型的定义
枚举顾名思义就是一一列举。把可能的取值一一列举。比如我们现实生活中:一周的星期一到星期日是有限的7天,可以一一列举。
enum Day 就是枚举类型。大括号里面是可能的取值,中间用逗号隔开,这与结构体类型的声明是不同的,结构体类型大括号里面是成员变量,用分号隔开。在使用枚举类型创建变量的时候就只能赋值成大括号里面的某个内容,比如enum Day x=Mon;
注:只有能一一列举的值才使用枚举类型,像身高体重这种可能的取值太多了,是不可能使用枚举类型的。
大括号里面列举的内容都是有值的,默认是从零开始,当然也可以在定义的时候进行赋值,如图所示
2.枚举的优点
我们可以使用 #define 定义常量,为什么非要使用枚举?
枚举的优点:1. 增加代码的可读性和可维护性
2. 和#define定义的标识符比较枚举有类型检查,更加严谨。
3. 防止了命名污染(封装)
4. 便于调试
5. 使用方便,一次可以定义多个常量
下面是一个枚举使用的例子
实际上这个GREEN就是2,但是我们不能直接把2赋给clr,增加了代码的可读性。
联合(共同体)
联合也是一种特殊的自定义类型这种类型定义的变量也包含一系列的成员,特征是这些成员公用同一块空间(所以联合也叫共用体)。
1.联合体类型的声明以及变量定义
2.联合体的特点
通过打印地址我们发现c和i的地址居然是一样的,这就说明c和i共用了同一块内存单元。这样虽然节省了空间,但是会有其他的问题,比如我在修改c的同时必然会把i也修改了,因此对于联合类型来讲,只能用其中的某一个成员变量。
利用联合体判断当前机器是大端还是小端
我们把i初始化为1,如果是小端,那么这四个字节存的就是01 00 00 00(16进制),如果是大端,这四个字节就是00 00 00 01,也就是说我们只需要判断第一个字节的内容即可,由于i和c又共用一块内存,因此在初始化i的时候c也会被初始化,c的内容就是第一个字节单元的内容,小端机器,高位在高地址,第一个字节单元应该是01,大端则是00,由此我们可以判断出当前机器是大端存储还是小端存储。
注:也可以直接创建一个int类型的变量a,初始化为1,然后
*(char*)&a判断是0还是1
3.联合体大小的计算
联合的大小至少是最大成员的大小。当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。
union Un1中第一个成员变量占5个字节,对齐数是1,第二个成员变量占4个字节,对齐数是4。
根据上面的规则,联合体大小最小的是最大成员变量所占字节大小,也就是5,但是5并不是最大对齐数也就是4的整数倍,离着最近的4的倍数是8,因此这个联合体union Un1的大小是8
union Un2第一个成员变量占14个字节,对齐数是2,第二个成员变量占4个字节,对齐数是4,。
union Un2中最大成员变量所占字节为14,最大对齐数是4,而14不是4的整数倍,离着最近的倍数是16,因此union Un2类型的大小是16个字节。
相关文章:

自定义类型详解 ----结构体,位段,枚举,联合
目录 结构体 1.不完全声明 2.结构体的自引用 3.定义与初始化 4.结构体内存对齐与结构体类型的大小 结构体嵌套问题 位段 1.什么是位段? 2.位段的内存分配 枚举 1.枚举类型的定义 2.枚举的优点 联合(共同体) 1.联合体类型的声明以…...

VueCLI核心知识综合案例TodoList
目录 1 拿到一个功能模块首先需要拆分组件: 2 使用组件实现静态页面的效果 3 分析数据保存在哪个组件 4 实现添加数据 5 实现复选框勾选 6 实现数据的删除 7 实现底部组件中数据的统计 8 实现勾选全部的小复选框来实现大复选框的勾选 9 实现勾选大复选框来…...

关于cuda路径问题
问题:Could not load dynamic library ‘libcudart.so.11.0’ 原因:调用系统环境下的cuda但系统环境没有装cuda 解决: 1.在系统环境装cuda,但如果每权限就不好操作; 2.用虚拟环境装好的cuda路径丢给环境变量 暂时性&am…...

六、Spring/Spring Boot整合ActiveMQ
Spring/Spring Boot整合ActiveMQ 一、Spring整合ActiveMQ1.pom.xml2.Queue - 队列2.1 applicationContext.xml2.2 生产者2.3 消费者 3.Topic - 主题3.1 applicationContext.xml3.2 生产者3.3 消费者 4.消费者 - 监听器4.1 编写监听器类4.2 配置监听器4.3 生产者消费者一体 二、…...

树莓派4B(Raspberry Pi 4B)使用docker搭建springBoot/springCloud服务
树莓派4B(Raspberry Pi 4B)使用docker搭建springBoot/springCloud服务 前提:本文基于Ubuntu,Java8,SpringBoot 2.6.13讲解 准备工作 准备SpringBoot/SpringCloud项目jar包 用 maven 打包springBoot/springCloud项目&…...

数据库设计、JDBC、数据库连接池
数据库设计 数据库设计概念 数据库设计就是根据业务 系统的具体需求,结合我们所选用的DBMS,为这个业务系统构造出最优的数据存储模型。建立数据库中的表结构以及表与表之间的关联关系的过程。有哪些表?表里有哪些字段?表和表之间有什么关系? 数据库设计的步骤…...

SpringBoot实现OneDrive文件上传
SpringBoot实现OneDrive文件上传 源码 OneDriveUpload: SpringBoot实现OneDrive文件上传 获取accessToken步骤 参考文档:针对 OneDrive API 的 Microsoft 帐户授权 - OneDrive dev center | Microsoft Learn 1.访问Azure创建应用Microsoft Azure,使…...

C++初阶:容器适配器介绍、stack和queue常用接口详解及模拟实现
介绍完了list类的相关内容后:C初阶:适合新手的手撕list(模拟实现list) 接下来进入新的篇章,stack和queue的介绍以及模拟: 文章目录 1.stack的初步介绍2.stack的使用3.queue的初步介绍4.queue的使用5.容器适…...

GRUB and the Boot Process on UEFI-based x86 Systems
background info : BIOS and UEFI-CSDN博客 The UEFI-based platform reads the partition table on the system storage and mounts the EFI System Partition (ESP), a VFAT partition labeled with a particular globally unique identifier (GUID). The ESP contains EFI a…...

2.C语言——输入输出
1.字符输入输出函数 1.输入:getchar() 字面意思,接收单个字符,使用方法 char a; a getchar();实际上效果等同于char a; scanf("%c",&a);2.输出:putchar() 2.格式化输入输出函数 1.输入:scanf() 格式: scanf(“格式控制…...

MySQL篇之SQL优化
一、表的设计优化 表的设计优化(参考阿里开发手册《嵩山版》): 1. 比如设置合适的数值(tinyint int bigint),要根据实际情况选择。 2. 比如设置合适的字符串类型(char和varchar)…...

QGis —— 1、Windows10下载安装QGis及插件
QGis官网 QGIS(自由开源的地理信息系统)是一个专业的GIS应用程序,它建立在免费和开源软件(FOSS)之上,并为此而自豪。QGIS 是一个方便使用的开源地理信息系统 (GIS),根据 GNU 通用公共许可授权。…...

【打工日常】使用docker部署Dashdot工具箱
一、Dashdot介绍 dashdot是一个简洁清晰的服务器数据仪表板,基于React实现 ,主要是显示操作系统、进程、存储、内存、网络这五个的数据。 二、本次实践介绍 1. 本次实践简介 本次实践部署环境为个人测试环境 2. 本地环境规划 本次实践环境规划…...

使用client-only 解决组件不兼容SSR问题
目录 前言 一、解决方案 1.基于Nuxt 框架的SSR应用 2.基于vue2框架的应用 3.基于vue3框架的应用 二、总结 往期回顾 前言 最近在我的单页面SSR应用上开发JSON编辑器功能,在引入组件后直接客户端跳转OK,但是在直接加载服务端渲染的时候一直报这…...

基于Java SSM框架实现网上报名系统项目【项目源码+论文说明】
基于java的SSM框架实现网上报名系统演示 摘要 随着互联网时代的到来,同时计算机网络技术高速发展,网络管理运用也变得越来越广泛。因此,建立一个B/S结构的网上报名系统,会使网上报名系统工作系统化、规范化,也会提高网…...

7.1 Qt 中输入行与按钮
目录 前言: 技能: 内容: 参考: 前言: line edit 与pushbotton的一点联动 当输入行有内容时,按钮才能使用,并能读出输入行的内容 技能: pushButton->setEnabled(false) 按钮不…...

云计算基础-网络虚拟化
虚拟交换机 什么是虚拟交换机 虚拟交换机是一种运行在虚拟化环境中的网络设备,其运行在宿主机的内存中,通过软件方式在宿主机内部实现了部分物理交换机的功能,如 VLAN 划分、流量控制、QoS 支持和安全功能等网络管理特性 虚拟交换机在云平…...

166基于matlab的通过峭度指标与互相关系数筛选IMF进行SVD分解去噪
基于matlab的通过峭度指标与互相关系数筛选IMF进行SVD分解去噪,分辨虚假imf,提取最大峭度imf图。输出去噪前后时域及其包络谱结果。程序已调通,可直接运行。 166 matlab SVD去噪 IMF筛选 包络谱 (xiaohongshu.com)...

第六十三天 服务攻防-框架安全CVE复现DjangoFlaskNode.JSJQuery
第六十三天 服务攻防-框架安全&CVE复现&Django&Flask&Node.JS&JQuery 知识点: 中间件及框架列表: IIS,Apache,Nginx,Tomcat,Docker,K8s,Weblogic.JBoos,WebSphere, Jenkins,GlassFish,Jetty,Jira,Struts2,Laravel,Solr,Shiro,Thin…...

最大子序和+旅行问题——单调队列
一、最大子序和 输入一个长度为 n 的整数序列,从中找出一段长度不超过 m 的连续子序列,使得子序列中所有数的和最大。 注意: 子序列的长度至少是 1。 输入 第一行输入两个整数 n,m (1 ≤ n,m ≤ 300000)。 第二行输入 n 个数,代…...

Unity设备分级策略
Unity设备分级策略 前言 之前自己做的设备分级策略,在此做一个简单的记录和思路分享。希望能给大家带来帮助。 分级策略 根据拟定的评分标准,预生成部分已知机型的分级信息,且保存在包内;如果设备没有被评级过,则优…...

自己在开发AI应用的过程总结的 Prompt - 持续更新
自己在开发AI应用的过程总结的 Prompt - 持续更新 0. 引言1. 让模型以"中文"进行回复2. 控制模型仅输出"hi"3. 让模型"提供简单、清晰而具体的回答"4. 让模型"在最后说谢谢" 0. 引言 我想,我们多半有着相似的经历…...

STM32——OLED菜单
文章目录 一.补充二. 二级菜单代码 简介:首先在我的51 I2C里面有OLED详细讲解,本期代码从51OLED基础上移植过来的,可以先看完那篇文章,在看这个,然后按键我是用的定时器扫描不会堵塞程序,可以翻开我的文章有单独的定时…...

Open CASCADE学习|布尔运算后消除内部拓扑
在CAD建模中,布尔运算是一种逻辑运算方法,通过这种方法,可以创建、修改或组合几何对象。布尔运算主要包括并集(UNION)、交集(INTERSECT)和差集(SUBTRACT)三种运算。 并集…...

【数据仓库】主题域和数据域
数据域与主题域区别 https://www.cnblogs.com/datadance/p/16898254.html 数据域是自下而上,以业务数据视角来划分数据,一般进行完业务系统数据调研之后就可以进行数据域的划分。针对公共明细层(DWD)进行主题划分。主题域则自上而…...

C#,二分法(Bisection Method)求解方程的算法与源代码
1 二分法 二分法是一种分治算法,是一种数学思维。 对于区间[a,b]上连续不断且f(a)f(b)<0的函数yf(x),通过不断地把函数f(x)的零点所在的区间…...

Portainer安装/快速上手
前置: 管理docker容器的工具 Portainer: Container Management Software for Kubernetes and Docker https://docs.portainer.io/v/ce-2.9/start/install/server/docker/linux 官网安装教程 Install Portainer CE with Docker on Linux - Portainer Documentat…...

恢复被.target勒索病毒加密的数据文件:拒绝向.target勒索病毒支付赎金
引言: 在当今数字时代,勒索病毒已成为网络安全领域的一大威胁,而.target勒索病毒是其中引起广泛关注的一种变种。本文将深入探讨.target勒索病毒的特点以及被其加密的数据文件恢复方法。数据的重要性不容小觑,您可添加我们的技术…...

【Linux网络编程六】服务器守护进程化Daemon
【Linux网络编程六】服务器守护进程化Daemon 一.背景知识:前台与后台二.相关操作三.Linux的进程间关系四.自成会话五.守护进程四步骤六.服务器守护进程化 一.背景知识:前台与后台 核心知识就是一个用户在启动Linux时,都会给一个session会话&a…...

MySQL之json数据操作
1 MySQL之JSON数据 总所周知,mysql5.7以上提供了一种新的字段格式json,大概是mysql想把非关系型和关系型数据库一口通吃,所以推出了这种非常好用的格式,这样,我们的很多基于mongoDB的业务都可以用mysql去实现了。当然…...