【自定义类型】--- 位段、枚举、联合


  • 💓博客主页:江池俊的博客
  • ⏩收录专栏:C语言进阶之路
  • 👉专栏推荐:✅C语言初阶之路 ✅数据结构探索
  • 💻代码仓库:江池俊的代码仓库
  • 🎉欢迎大家点赞👍评论📝收藏⭐

在这里插入图片描述

文章目录

  • 一、✨结构体的那些事✨
    • 1.1 结构体的自引用
    • `1.2 结构体内存对齐`
      • 【结构体嵌套问题】
      • 为什么存在内存对齐?
    • 1.3 offsetof宏
  • 二、🍁位段(Bit-fields)🍁
    • 2.1 什么是位段?
    • 2.2 定义位段
    • 2.3 位段的内存分配
    • 2.4 位段的跨平台问题
    • 2.5 位段的用途
  • 三、👀枚举👀
    • 3.1 枚举类型的定义
    • 3.2 枚举的优点
    • 3.3 枚举的使用
  • 四、 💫联合(共用体)💫
    • 4.1 联合类型的定义
    • 4.2 联合的特点
    • 4.3 联合大小的计算

前言

上节【C语言】结构体解谜:拆解数据的力量!已经讲解了结构体这种自定义类型,那么接下来我将带着大家一起深入结构体以及其他自定义类型的学习。

一、✨结构体的那些事✨

概念: 结构是一些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量。

1.1 结构体的自引用

在C语言中,结构体内部是不能包含直接指向自己类型的成员的,这会导致无法确定结构体的大小。这是因为一个结构体的大小必须是确定的,以便程序在内存中正确地分配空间。

如果需要在结构体内部包含指向相同类型的成员,可以使用指向结构体的指针来实现这个目的。

以下是一个正确的示例:

struct Node
{int data;struct Node* next; // 使用指针指向相同类型的结构体
};

在这个例子中,next 成员是一个指向 struct Node 类型的指针,而不是直接嵌套一个 struct Node 类型。

然而,对于以下代码:

struct Node
{int data;struct Node next; // 这种方式是错误的
};

这是不合法的,因为 struct Node 中包含了一个直接指向相同类型的成员 next,这将导致无法确定结构体的大小。

1.2 结构体内存对齐

我们已经掌握了结构体的基本使用了。
现在我们深入讨论一个问题:计算结构体的大小。
这也是一个特别热门的考点: 结构体内存对齐

在C语言中,结构体内存对齐是指编译器如何在内存中分配结构体的成员以保证存取效率和对齐要求。它的目的是为了优化内存访问的性能。

那么结构体大小我们将如何计算呢?

首先得掌握结构体的对齐规则:

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

  2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
    对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值。
    VS中默认的值为8

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

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

  5. sizeof操作符sizeof 结构体会返回整个结构体的大小,包括填充字节。

  6. #pragma pack(n)#pragma pack 是一个预处理指令,可以用来改变默认的对齐方式。#pragma pack(n) 将指定对齐值为 n 字节。这个指令在一些特殊情况下可能会用到,但一般情况下不建议随意修改对齐方式。

例子:

struct Example {char a;    // 占用1字节int b;     // 在32位系统下通常占用4字节double c;  // 通常占用8字节
}; // 在32位系统下,这个结构体的大小为 16 字节(包括填充)

在这里插入图片描述

【结构体嵌套问题】

struct S3
{double d;char c;int i;
};
printf("%d\n", sizeof(struct S3)); //结果是 16 
//结构体嵌套问题
struct S4
{char c1;struct S3 s3;double d;
};
printf("%d\n", sizeof(struct S4)); //结果是 32

在这里插入图片描述
需要强调的是,具体的对齐规则和表现可能会因编译器、编译器版本和目标平台的不同而略有差异,因此在编写特定平台下对齐要求敏感的代码时,最好查阅相应的编译器文档或规范。

为什么存在内存对齐?

大部分的参考资料都是如是说的:

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

总体来说:

结构体的内存对齐是拿 空间 来换取 时间 的做法。

那在设计结构体的时候,我们既要满足对齐,又要节省空间,如何做到:

让占用空间小的成员尽量集中在一起。

//例如:
struct S1
{char c1;int i;char c2;
};//大小为 12
struct S2
{char c1;char c2;int i;
};//大小为 8

S1和S2类型的成员一模一样,但是S1和S2所占空间的大小有了一些区别。

1.3 offsetof宏

头文件: <stddef.h>
声明

offsetof(type, member-designator);

参数

  • type - - - 这是一个 class 类型,其中,member-designator 是一个有效的成员指示器。
  • member-designator - - - 这是一个 class 类型的成员指示器。

返回值

  • 该宏返回类型为 size_t 的值,表示 type 中成员的偏移量。

C 库宏 offsetof(type, member-designator) 会生成一个类型为 size_t 的整型常量,它是一个结构成员相对于结构开头的字节偏移量。成员是由 member-designator 给定的,结构的名称是在 type 中给定的。

【实例】:


#include<stdio.h>
#include<stddef.h>
struct S3
{double d;char c;int i;
}; //大小为16
//结构体嵌套问题
struct S4
{char c1;struct S3 s3;double d;
}; //大小为32
int main()
{printf("aS3 结构中的 c 偏移 = %d 字节。\n",offsetof(struct S3, c)); //结果是 8printf("S4 结构中的 c1 偏移 = %d 字节。\n",offsetof(struct S4, c1)); //结果是 0printf("S4 结构中的 s3 偏移 = %d 字节。\n",offsetof(struct S4, s3)); //结果是 8 return 0;
}

在这里插入图片描述
在这里插入图片描述


二、🍁位段(Bit-fields)🍁

在C语言中,位段(Bit-fields)是一种非常有用的数据结构,允许你定义数据成员占用的位数,从而有效地利用存储空间。位段特别适用于需要精细控制内存占用和性能的情况。在本文中,我们将深入研究C语言中的位段,包括如何定义、使用和优化它们。

2.1 什么是位段?

位段是一种结构体成员,它允许你定义成员占用的位数。这样,你可以在一个字节(或更大的内存单元)中将不同的位用于不同的数据,而不是整个字节。这有助于节省内存,并在某些情况下提高性能。

2.2 定义位段

位段的声明和结构是类似的,有两个不同:

  1. 位段的成员必须是 intunsigned intsigned int
  2. 位段的成员名后边有一个冒号和一个数字。

在结构体中定义位段的语法如下:

struct MyStruct {type memberName : numberOfBits;
};
  • type 是成员的数据类型,通常是 intunsigned int 或其他整数类型。
  • memberName 是成员的名称。
  • numberOfBits 指定了成员占用的位数。

2.3 位段的内存分配

使用位段时需要注意以下事项:

  1. 位段的成员可以是 intunsigned intsigned int 或者是 char (属于整形家族)类型
  2. 位段的总位数不能超过成员的类型的总位数。
  3. 位段成员不能取地址,也不能作为函数参数传递。
  4. 位段的空间上是按照需要以 4个字节( int ) 或者 1个字节( char ) 的方式来开辟的。
  5. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。
//一个例子
struct S
{char a : 3;char b : 4;char c : 5;char d : 4;
};
struct S s = { 0 };
s.a = 10;
s.b = 12;
s.c = 3;
s.d = 4;
//空间是如何开辟的?

在这里插入图片描述

2.4 位段的跨平台问题

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

总结:

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

2.5 位段的用途

  1. 节省内存: 位段允许你精确地控制每个成员占用的位数,从而节省内存。这对于嵌入式系统和大规模数据结构特别有用。

  2. 提高性能: 在某些情况下,位段可以提高内存访问性能,因为它们可以减少数据传输的时间。

  3. 代码可读性: 通过使用位段,你可以更清晰地表示特定的标志或位字段,从而提高代码的可读性。

在这里插入图片描述


三、👀枚举👀

枚举顾名思义就是一一列举。
把可能的取值一一列举。
比如我们现实生活中:

一周的星期一到星期日是有限的7天,可以一一列举。
性别有:男、女、保密,也可以一一列举。
月份有12个月,也可以一一列举

这里就可以使用枚举了。

3.1 枚举类型的定义

enum Day//星期
{Mon,Tues,Wed,Thur,Fri,Sat,Sun
};
enum Sex//性别
{MALE,FEMALE,SECRET
};
enum Color//颜色
{RED,GREEN,BLUE
};

以上定义的 enum Dayenum Sexenum Color 都是枚举类型。
{}中的内容是枚举类型的可能取值,也叫 枚举常量

这些可能取值都是有值的,默认从0开始一次递增1,当然在定义的时候也可以赋初值。
例如:

enum Color//颜色
{RED = 1,GREEN = 2,BLUE = 4
};

3.2 枚举的优点

为什么使用枚举?

我们可以使用 #define 定义常量,为什么非要使用枚举?
枚举的优点:

  1. 增加代码的可读性和可维护性
  2. 和#define定义的标识符比较枚举有类型检查,更加严谨。
  3. 防止了命名污染(封装)
  4. 便于调试
  5. 使用方便,一次可以定义多个常量

3.3 枚举的使用

enum Color//颜色
{RED = 1,GREEN = 2,BLUE = 4
};
int main()
{enum Color clr = GREEN;//只能拿枚举常量给枚举变量赋值,才不会出现类型的差异。clr = 5; //赋值号左边是枚举类型,右边是整型,若在c++中是不能赋值的return 0;
}

四、 💫联合(共用体)💫

联合体(Union)是一种特殊的数据类型,允许在相同的内存位置存储不同类型的数据。它类似于结构体,但是不同的是,联合体的成员共享同一块内存空间。
关键字:union

4.1 联合类型的定义

联合也是一种特殊的自定义类型
这种类型定义的变量也包含一系列的成员,特征是 这些成员公用同一块空间(所以联合也叫共用体)。
比如:

#include<stdio.h>
//联合类型的声明
union Un
{char c;int i;
};int main()
{//联合变量的定义union Un un;//计算连个变量的大小printf("%d\n", sizeof(un)); //结果为 4return 0;
}

在这里插入图片描述

4.2 联合的特点

联合的成员是共用同一块内存空间的,这样一个联合变量的大小,至少是最大成员的大小(因为联合至少得有能力保存最大的那个成员)。

面试题:

判断当前计算机的大小端存储

//判断当前计算机的大小端存储
#include<stdio.h>check_sys()
{union Un{char c;int i;}u;u.i = 1;return u.c;//返回1表示小端,返回0表示大端
}int main()
{if (check_sys())printf("小端\n");elseprintf("大端\n");return 0;
} //vs2022下输出结果为:小端

在这里插入图片描述

4.3 联合大小的计算

  • 联合的大小至少是最大成员的大小。
  • 当最大成员大小不是最大对齐数的整数倍的时候,就要对齐到最大对齐数的整数倍。

比如:

#include<stdio.h>
union Un1
{char c[5];int i;
};
union Un2
{short c[7];int i;
};
int main()
{//下面输出的结果是什么?printf("%d\n", sizeof(union Un1)); //结果为 8printf("%d\n", sizeof(union Un2)); //结果为 16return 0;
}

在这里插入图片描述


今天的分享就到这里了,感谢你的阅读,如果你有任何疑问或者想分享你的经验,请在下方留言,我会非常乐意与你讨论。

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

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

相关文章

React18+Ts项目配置husky、eslint、pretttier、commitLint

前言 我的项目版本如下&#xff1a; React&#xff1a; V18.2.0Node.js: V16.14.0TypeScript&#xff1a;最新版工具&#xff1a; VsCode 本文将采用图文详解的方式&#xff0c;手把手带你快速完成在React项目中配置husky、prettier、commitLint&#xff0c;实现编码规范的统…

使用sqlmap获取数据步骤

文章目录 1.使用sqlmap获取所有数据库2.使用sqlmap获取当前连接数据库3.使用sqlmap获取当前数据库下所有表名4.使用sqlmap获取当前数据库下某个表下所有列名5.使用sqlmap获取当前数据库下某个表下指定字段的数据6.测试当前用户是否是管理员7.使用burpsqlmap批量检测8.脱库命令9…

算法竞赛备赛之贪心算法训练提升,贪心算法基础掌握

1.区间问题 905.区间选点 给定N个闭区间[ai, bi]&#xff0c;请你在数轴上选择尽量少的点&#xff0c;使得每个区间内至少包含一个选出的点。 输出选择的点的最小数量&#xff0c;位于区间端点上的点也算作是区间内。 将每个按区间的右端点从小到大排序 从前往后依次枚举每…

记录:Unity脚本的编写

目录 前言添加脚本到unity编写c#脚本查看效果 前言 在学习软件构造这门课的时候&#xff0c;对unity和c#进行了 一定程度的学习&#xff0c;包括简单的建立地形&#xff0c;添加对象&#xff0c;添加材质等&#xff0c;前不久刚好学习了如何通过c#脚本对模型进行操控&#xff…

五、2023.10.1.C++stl.5

文章目录 65、请说说 STL 的基本组成部分?66、请说说 STL 中常见的容器&#xff0c;并介绍一下实现原理&#xff1f;67、请说说 STL 中常见的容器&#xff0c;并介绍一下实现原理&#xff1f;68、请你来介绍一下 STL 的空间配置器&#xff08;allocator&#xff09;&#xff1…

分布式并行训练(DP、DDP、DeepSpeed)

[pytorch distributed] 01 nn.DataParallel 数据并行初步 数据并行 vs. 模型并行 数据并行&#xff1a;模型拷贝&#xff08;per device&#xff09;&#xff0c;数据 split/chunk&#xff08;对batch切分&#xff09; 每个device上都拷贝一份完整模型&#xff0c;每个device分…

密码技术 (5) - 数字签名

一. 前言 前面在介绍消息认证码时&#xff0c;我们知道消息认证码虽然可以确认消息的完整性&#xff0c;但是无法防止否认问题。而数字签名可以解决否认的问题&#xff0c;接下来介绍数字签名的原理。 二. 数字签名的原理 数字签名和公钥密码一样&#xff0c;也有公钥和私钥&am…

字符串函数(一)

✨博客主页&#xff1a;小钱编程成长记 &#x1f388;博客专栏&#xff1a;进阶C语言 字符串函数&#xff08;一&#xff09; 0.前言1.求字符串长度的函数1.1 strlen&#xff08;字符串长度&#xff09; 2.长度不受限制的字符串函数2.1 strcpy&#xff08;字符串拷贝&#xff0…

直播协议 python 常见直播协议

1. 推流、直播 和 点播分别是什么意思&#xff1f; 推流 主播将本地视频源和音频源推送到云服务器&#xff0c;也被称为“RTMP发布”。 直播 即直接观看主播实时推送过来的音视频数据。 点播 视频源已经事先存储于云服务器之上的音视频文件&#xff0c;观众随时可以观看。 目…

怒刷LeetCode的第22天(Java版)

目录 第一题 题目来源 题目内容 解决方法 方法一&#xff1a;回溯算法 方法二&#xff1a;基于位运算的回溯 第二题 题目来源 题目内容 解决方法 方法一&#xff1a;动态规划 方法二&#xff1a;分治法 方法三&#xff1a;前缀和数组 第三题 题目来源 题目内容…

Acwing 842. 排列数字

Acwing 842. 排列数字 知识点题目描述思路讲解代码展示 知识点 DFS 题目描述 思路讲解 DFS重点是&#xff1a;顺序&#xff01;&#xff08;暴力的手法&#xff09;&#xff08;递归&#xff09; 用 path 数组保存排列&#xff0c;当排列的长度为 n 时&#xff0c;是一种方…

【Leetcode】 17. 电话号码的字母组合

给定一个仅包含数字 2-9 的字符串&#xff0c;返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下&#xff08;与电话按键相同&#xff09;。注意 1 不对应任何字母。 示例 1&#xff1a; 输入&#xff1a;digits "23" 输出&…

Java笔记五(数组)

目录 数组 数组声明创建 数组初始化的三种初始化类型&#xff1a; 静态初始化 动态初始化 数组的默认初始化 数组的四个基本特点 数组边界 数组的使用 示例一&#xff1a;计算所有的元素和以及查找最大元素 示例二&#xff1a;For-Each循环 示例三&#xff1a;数组作…

Ubuntu 安装 Docker 的详细步骤

文章目录 简介1.更新2.安装必要的软件包2.1 基于阿里源 3.验证 Docker 安装是否成功4.安装后的一些常规设置及常用的命令4.1 启动 Docker4.2 Docker 在系统启动时自动运行4.3 运行一个 Hello World 镜像4.4 查看docker运行状态 欢迎来到这篇关于在 Ubuntu 上安装 Docker 的教程…

鞋类 整鞋试验方法 剥离强度

声明 本文是学习GB-T 3903.3-2011 鞋类 整鞋试验方法 剥离强度. 而整理的学习笔记,分享出来希望更多人受益,如果存在侵权请及时联系我们 1 范围 GB/T 3903 的本部分规定了整鞋鞋底与鞋帮或外底与外中底之间剥离强度的试验方法。 本部分适用于采用模压、硫化、注塑、灌注、胶…

【c++随笔07】常量、变量、static

【c随笔07】常量、变量、static 1、常量、变量1.1、声明变量1.2、使用常量 2、static介绍2.1、static 局部变量2.2、static 全局变量2.3、C static静态成员变量2.4、C static静态成员函数详解 原创地址&#xff0c;https://zhengjunxue.blog.csdn.net/article/details/13167770…

【数据结构】——顺序表详解

大家好&#xff01;当我们学习了动态内存管理后&#xff0c;就可以写一个管理数据的顺序表了&#xff01;&#xff01;&#xff01; 顺序表的理解&#xff1a; 线性表是最基本、最简单、也是最常用的一种数据结构。线性表&#xff08;linear list&#xff09;是数据结构的一种…

02-Zookeeper实战

上一篇&#xff1a;01-Zookeeper特性与节点数据类型详解 1. zookeeper安装 Step1&#xff1a; 配置JAVA环境&#xff0c;检验环境&#xff1a; java -versionStep2: 下载解压 zookeeper wget https://mirror.bit.edu.cn/apache/zookeeper/zookeeper-3.5.8/apache-zookeepe…

响应式设计的实现方式

一. 什么是响应式 响应式网站设计是一种网络页面设计布局。页面的设计与开发应当根据用户行为以及设备环境&#xff08;系统平台&#xff0c;屏幕尺寸&#xff0c;屏幕定向等&#xff09;进行相应的响应和调整。 响应式网站常见特点&#xff1a; 1. 同时适配PC平板手机。 2…

Win10自带输入法怎么删除-Win10卸载微软输入法的方法

Win10自带输入法怎么删除&#xff1f;Win10系统自带输入法就是微软输入法&#xff0c;这个输入法满足了很多用户的输入需求。但是&#xff0c;有些用户想要使用其它的输入法&#xff0c;这时候就想删除掉微软输入法。下面小编给大家介绍最简单方便的卸载方法吧。 Win10卸载微软…