C语言进阶第八篇【三大自定义类型详解:结构体struct,枚举enum,联合union】(上)

简介: C语言进阶第八篇【三大自定义类型详解:结构体struct,枚举enum,联合union】(上)

前言:Hello大家好,我是@每天都要敲代码!今天就带大家学习一下新的内容;三大自定义类型:结构体struct,枚举enum,联合union;看着内容很少,其实知识点也很丰富;希望下面我的讲解能让大家有所收获!


1. 结构体

1.1 结构的基础知识和声明

结构是一些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型的变量。我们需要掌握:结构体类型的声明、结构的自引用、结构体变量的定义和初始化、结构体内存对齐、结构体传参、结构体实现位段(位段的填充&可移植性)!


回顾:数组是一组相同类型的元素的集合!


对于结构体的声明,我们就拿一个例子来解释;更加的容易理解:


❤️例:

cdc7fac470014598b738933bf6d5eff3.png



1.2 匿名结构体的声明

匿名结构体的声明是指在声明结构的时候,可以不完全的声明!还是不明白什么意思?我们不妨拿个例子来理解:


❤️例:

79a2e005ed934028b72eb386dffc07f4.png


1.3 结构的自引用

❤️1、一个结构体引用另一个结构体

fac2fbdcd31a455ebb6bcb11330372e7.png


一个结构体引用另一个结构体是完全没问题的!


❤️2、自己的结构体引用自己


fc76fdd352234bad9be3b59a3080d855.png


结构体里面,自己引用自己,会造成死循环;相当于死递归,是错误的引用!


❤️3、正确的自引用方法


fe6866e0921044f597810a2cf1031d70.png


一个结构体不是包含同类型结构体的变量,而是包含同类型结构体的指针,是完全没问题的!


1.4 结构体变量的初始化和打印

❤️例:

82ee67785ad9455bbac849e6edf095e1.png



1.5 结构体内存对齐(结构体大小的计算)


首先先介绍一下对齐规则:


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

(2)其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。


        对齐数 = 编译器默认的一个对齐数 与 该成员大小的较小值。VS中默认的值为8

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

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


tips:一定要先理解上面的四句话,在进行习题的练习!


❤️例题1:


10be0e11734143ada4a61dfa09e995bb.png


⭐️对于结构体S:


(1) c1放在偏移量为0处,且占一个字节!


(2)i占4个字节,直接从偏移量为1处开始?当然不是!其他成员变量要对齐到某个(对齐数)的整数倍的地址处。对齐数 = VS中默认的值为8与 该成员大小的较小值。所以对齐数应该是4,应该从4的倍数开始,要舍去1、2、3这三个字节!最终从4开始:4、5、6、7;如下图:


a5a61f28cf004573bd32319b55d134ac.png


(3)最终结构体的大小,也要是这个结构体成员变量的对齐数最大值的整数倍!这里也就是4的倍数;结合(2)最终结果就是8!


⭐️对于结构体S2:


(1)c1放在偏移量为0处,且占一个字节!


(2)i占4个字节,直接从偏移量为1处开始?当然不是!其他成员变量要对齐到某个(对齐数)的整数倍的地址处。对齐数 = VS中默认的值为8与 该成员大小的较小值。所以对齐数应该是4,应该从4的倍数开始,要舍去1、2、3这三个字节!最终从4开始:4、5、6、7;


(3)c2占一个字节,对齐数 = VS中默认的值为8与 该成员大小的较小值。所以对齐数应该是1;刚好放到8的位置;如下图:


9550b848207e4fe6ac7c5e79bc5298c3.png


(4)(3)最终结构体的大小,也要是这个结构体成员变量的对齐数最大值的整数倍!这里也就是4的倍数;结合(2)(3)最终结果就是:12;而不是9(9不是4的倍数);所以最终还是会浪费3个字节9、10、11!


❤️例题2:

05ee9e61abd340d994ca1cc0dcb17759.png



⭐️对于结构体S:


2abb066bd0524c928e10b9bf60a600f0.png


通过画图我们知道,S的大小应该是9,但是9却不是4的倍数;所以最终结果就是12!


⭐️对于结构体S2:


14087a1569594a1f847af7fdf2c6fba8.png


通过画图我们知道,S2的大小应该是16,16刚好是4的倍数;所以最终结果就是16!


⭐️对于结构体S3:


734e4b0ab1db46ddb47de90276cc67f6.png


通过画图我们知道,S3的大小应该是8,8刚好是4的倍数;所以最终结果就是8!


⭐️对于结构体S4:


792f4ec211a94ea4be760ac981ff24ba.png


通过画图我们知道,S4的大小应该是16,16刚好是8的倍数;所以最终结果就是16!


⭐️对于结构体S5:

fa84b32c7e0c4f06ada33133a7a428a8.png



对于结构体S5;里面嵌套着结构体S4;首先是c1在偏移量0处,且占1个字节!然后是结构体S4我们已经计算出来是16字节的大小;这就占17个字节就是0-16,下一个位就是17开始;最后d占8个字节,17不是8的倍数,应从8的倍数24开始,数8个就是到31;所以最终结果就是32!


补充:


(1)在设计结构体的时候,既要满足对齐,也要节省空间,怎么做到?

答:让占用空间小的成员尽量集中到一起!


(2)为什么存在内存对齐?


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

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

     2. 性能原因:

数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。

相关文章
|
22天前
|
存储 C语言
如何在 C 语言中实现结构体的深拷贝
在C语言中实现结构体的深拷贝,需要手动分配内存并逐个复制成员变量,确保新结构体与原结构体完全独立,避免浅拷贝导致的数据共享问题。具体方法包括使用 `malloc` 分配内存和 `memcpy` 或手动赋值。
30 10
|
22天前
|
安全 编译器 Linux
【c语言】轻松拿捏自定义类型
本文介绍了C语言中的三种自定义类型:结构体、联合体和枚举类型。结构体可以包含多个不同类型的成员,支持自引用和内存对齐。联合体的所有成员共享同一块内存,适用于判断机器的大小端。枚举类型用于列举固定值,增加代码的可读性和安全性。文中详细讲解了每种类型的声明、特点和使用方法,并提供了示例代码。
21 3
|
22天前
|
存储 大数据 编译器
C语言:结构体对齐规则
C语言中,结构体对齐规则是指编译器为了提高数据访问效率,会根据成员变量的类型对结构体中的成员进行内存对齐。通常遵循编译器默认的对齐方式或使用特定的对齐指令来优化结构体布局,以减少内存浪费并提升性能。
|
26天前
|
编译器 C语言
共用体和结构体在 C 语言中的优先级是怎样的
在C语言中,共用体(union)和结构体(struct)的优先级相同,它们都是用户自定义的数据类型,用于组合不同类型的数据。但是,共用体中的所有成员共享同一段内存,而结构体中的成员各自占用独立的内存空间。
|
26天前
|
存储 C语言
C语言:结构体与共用体的区别
C语言中,结构体(struct)和共用体(union)都用于组合不同类型的数据,但使用方式不同。结构体为每个成员分配独立的内存空间,而共用体的所有成员共享同一段内存,节省空间但需谨慎使用。
|
1月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
34 3
|
7天前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
22 6
|
26天前
|
存储 缓存 C语言
【c语言】简单的算术操作符、输入输出函数
本文介绍了C语言中的算术操作符、赋值操作符、单目操作符以及输入输出函数 `printf` 和 `scanf` 的基本用法。算术操作符包括加、减、乘、除和求余,其中除法和求余运算有特殊规则。赋值操作符用于给变量赋值,并支持复合赋值。单目操作符包括自增自减、正负号和强制类型转换。输入输出函数 `printf` 和 `scanf` 用于格式化输入和输出,支持多种占位符和格式控制。通过示例代码详细解释了这些操作符和函数的使用方法。
34 10
|
20天前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。
|
25天前
|
机器学习/深度学习 C语言
【c语言】一篇文章搞懂函数递归
本文详细介绍了函数递归的概念、思想及其限制条件,并通过求阶乘、打印整数每一位和求斐波那契数等实例,展示了递归的应用。递归的核心在于将大问题分解为小问题,但需注意递归可能导致效率低下和栈溢出的问题。文章最后总结了递归的优缺点,提醒读者在实际编程中合理使用递归。
53 7