c语言结构体知识系统详解

简介: c语言结构体知识系统详解

本篇文章带来结构体相关知识与结构体内存对齐的规则详细讲解!

如果您觉得文章不错,期待你的一键三连哦,你的鼓励是我创作的动力之源,让我们一起加油,一起奔跑,让我们顶峰相见!!!

目录

编辑

本篇文章带来结构体相关知识与结构体内存对齐的规则详细讲解!

如果您觉得文章不错,期待你的一键三连哦,你的鼓励是我创作的动力之源,让我们一起加油,一起奔跑,让我们顶峰相见!!!

一.认识什么是结构体

二.结构体的声明

特殊的结构体申明

编辑

结构体类型重命名

三.结构体的自引用

匿名结构体自引用时的常见错误:

四.结构体变量的初始化

结构体嵌套结构体的初始化和打印

五.结构体内存对齐(重中之重)

重点:

结构体的对齐规则:

宏offsetof

结构体嵌套结构体的例子

为什么存在内存对齐?

修改默认对齐数

五.结构体传参


一.认识什么是结构体

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

这里和数组进行区分,数组:是一组 相同类型元素的集合;

二.结构体的声明

如:

tag                  是 结构体标签名 ,是可以自定义的;

member-list    是 成员列表

variable-list     是 变量列表

需要注意的是:后面的分号不能丢;

这里的 :

 struct   Stu          是结构体类型;

 char  name[ 20] , int age  ,char sex[5]   都是成员变量

 S1, S2, S3            是结构体变量

有两种方式创建结构体变量;

第一种:就是先进行结构体申明后,再创建结构体变量 (是全局变量) 如图左边;

第二种:在进行结构体申明是就直接在括号后面进行结构体创建(是局部变量)  如图右边;

特殊的结构体申明

注意:匿名结构体类型(不完全的声明),创建结构体变量时,只能用上面的第一种方式;(如图)

 上面的两个结构在声明的时候省略掉了结构体标签(tag);

那么,问题来了,上面代码中的   p=&S  时合法的吗?

警告:

编译器会把上面的两个声明当成完全不同的两个类型。

所以是非法的。

结构体类型重命名

如下代码:

上面代码:将结构体类型 struct  Stu  重命名为Stu;

也可以将匿名结构体类型重命名,如下面代码:

因此创建结构体变量时,就可以直接用 Stu   S1,S2,S3这种方式创建;

三.结构体的自引用

在结构中包含一个类型为该结构本身的成员是否可以呢?

结构体中包含一个结构体(自身或则另一个结构体),需要用到结构体指针;

如下代码:

匿名结构体自引用时的常见错误:

四.结构体变量的初始化

注意:

不使用点操作符,只能按照声明时成员列表的顺序进行初始化,使用点操作符可以改变初始的顺序

结构体嵌套结构体的初始化和打印

结构体是用大括号括起来的,所以当一个结构体里面含另一个结构体时,里面的结构体也需要用大括号;

打印上面结构体里面的数据:例如打印Sn2的数据

五.结构体内存对齐(重中之重)

重点:

特别热门的考点:    结构体内存对齐

考察方式:               计算结构体的大小

例:

但是实际运行的结果是:

这是为什么呢?这就讲到结构体的对齐规则了;

结构体的对齐规则:

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

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

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

VS中默认的值为8

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

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

4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整

体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。

再将上述代码进行分析:

宏offsetof

1.是用来计算结构体成员相较于结构体起始位置的偏移量;

2.offsetof需要包含的头文件是:stddef.h

我们用这个宏来验证我们的分析:

运行的 结果和上面我们分析对照可以发现是一样的;

结构体嵌套结构体的例子

打印来验证我们的分析:

为什么存在内存对齐?

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

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

2. 性能原因

数据结构 ( 尤其是栈 ) 应该尽可能地在自然边界上对齐。

原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访

问。

总体来说:

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

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

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

修改默认对齐数

#pragma 这个预处理指令,可以改变我们的默认对齐数。

示例:

结论:

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

五.结构体传参

结构体传参的时候,要传结构体的地址。

上面的 print1 和 print2 函数哪个好些?

答案是:首选print2函数

原因:

函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。

如果传递一个结构体对象的时候,结构体过大,参数压栈的的系统开销比较大,所以会导致性能的

下降。

结论:

结构体传参的时候,要传结构体的地址。

本章完~


目录
相关文章
|
2月前
|
存储 安全 数据管理
C语言之考勤模拟系统平台(千行代码)
C语言之考勤模拟系统平台(千行代码)
64 4
|
2月前
|
程序员 C语言 开发者
pymalloc 和系统的 malloc 有什么区别
pymalloc 和系统的 malloc 有什么区别
|
1月前
|
IDE 编译器 开发工具
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
在本文中,我们系统地讲解了常见的 `#pragma` 指令,包括其基本用法、编译器支持情况、示例代码以及与传统方法的对比。`#pragma` 指令是一个强大的工具,可以帮助开发者精细控制编译器的行为,优化代码性能,避免错误,并确保跨平台兼容性。然而,使用这些指令时需要特别注意编译器的支持情况,因为并非所有的 `#pragma` 指令都能在所有编译器中得到支持。
117 41
【C语言】全面系统讲解 `#pragma` 指令:从基本用法到高级应用
|
1月前
|
存储 编译器 C语言
【C语言】C语言的变量和声明系统性讲解
在C语言中,声明和定义是两个关键概念,分别用于告知编译器变量或函数的存在(声明)和实际创建及分配内存(定义)。声明可以多次出现,而定义只能有一次。声明通常位于头文件中,定义则在源文件中。通过合理组织头文件和源文件,可以提高代码的模块化和可维护性。示例包括全局变量、局部变量、函数、结构体、联合体、数组、字符串、枚举和指针的声明与定义。
53 12
|
1月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
154 14
|
1月前
|
存储 编译器 C语言
【C语言】结构体详解 -《探索C语言的 “小宇宙” 》
结构体通过`struct`关键字定义。定义结构体时,需要指定结构体的名称以及结构体内部的成员变量。
183 10
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
191 13
|
2月前
|
存储 数据建模 程序员
C 语言结构体 —— 数据封装的利器
C语言结构体是一种用户自定义的数据类型,用于将不同类型的数据组合在一起,形成一个整体。它支持数据封装,便于管理和传递复杂数据,是程序设计中的重要工具。
|
2月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
78 11
|
2月前
|
机器学习/深度学习 算法 数据挖掘
C语言在机器学习中的应用及其重要性。C语言以其高效性、灵活性和可移植性,适合开发高性能的机器学习算法,尤其在底层算法实现、嵌入式系统和高性能计算中表现突出
本文探讨了C语言在机器学习中的应用及其重要性。C语言以其高效性、灵活性和可移植性,适合开发高性能的机器学习算法,尤其在底层算法实现、嵌入式系统和高性能计算中表现突出。文章还介绍了C语言在知名机器学习库中的作用,以及与Python等语言结合使用的案例,展望了其未来发展的挑战与机遇。
56 1