【再识C进阶4】详细介绍自定义类型——结构体、枚举和联合

学习目标:

       在上一篇博客中,我们已经详细地学习了字符分类函数、字符转换函数和内存函数。那这一篇博客和上一篇博客的关系不是那么相连。

       这一篇博客主要介绍一下自定义类型,因为在解决实际问题时,由于世界上的因素有很多,我们需要建立不同的数据类型来描述这些变量,但是C语言本身创立的类型不是很多,所以需要我们用户自己根据需求进行创建,于是就有了这一篇博客!


学习内容:

通过上面的学习目标,我们可以列出要学习的内容:

  1. 结构体的相关知识
  2. 结构体的内存对齐
  3. 结构体实现位段
  4. 枚举的相关知识
  5. 联合的相关知识

一、结构体的相关知识

1.1 结构的基本知识

       通俗来讲,结构是一些值的集合,这些值称为结构成员读者们看到这句话是不是似曾相识啊,在前面的数组这一章,我们讲过数组的概念——是一些相同类型的值的集合。小编在这里提一嘴,是因为数组存放的值和今天我们讲述结构体存放的值是不一样的。结构体的成员可以是不同类型的变量,一定要区分清楚!

       因为结构体中,我们也可以放置相同类型的值的集合;而数组只能放置相同类型的值的集合。我们要好好利用结构体来解决实际生活中的问题。

1.2 结构体的声明

1.2.1 结构体的内容介绍

       在上图中,我们介绍了结构体类型中的各个位置的含义和概念,下面,我们来自己设计一个学生的结构体类型:

struct Student {char name[20];int age;char sex[5];double score;
};

1.2.2 结构体是如何创建变量 

       定义完结构体类型后,这就相当于又创建了一个数据类型,我们可以根据这个数据类型来创建一个或多个结构体变量,同时也可以创建一个结构体数组,这和基本数据类型大差不差。接下来,我们就来创建结构体变量:

1.2.3 利用typedef简化类型名字

       你们会不会因为结构体类型的名字比较长,而觉得很不方便,我们可以利用 typedef 来进行对类型的重命名:下面,我们来看一个例子:

1.3 特殊的声明

1.3.1 匿名结构体的概念和代码

       这个声明就像标题所说的一样,是有点特殊的,也是不常用的。我们在进行编写代码时,在设计结构体时,可以进行不完全的设计声明。因为编写结构体时没有给出相应的名字,所以这种设计的结构体叫做匿名结构体。具体例子我们看下面:

//匿名结构体类型
struct{int a;char b;float c;
}b;

1.3.2 匿名结构体和普通结构体的区别 

为了便于理解,我们在进行匿名结构体类型与普通结构体类型之间的区别(如下图)

1.3.3 有关匿名结构体的一道题目 

在了解完上面匿名结构体的概念后,我们来看这么一道题:

//匿名结构体类型
struct {int a;char b;float c;
}b;
struct {int a;char b;float c;
} *p;

上面两个结构体在声明的时候省略掉了结构体标签tag。那么请问:

//在上面的代码基础上,下面的代码合法吗?
p = &b;

答案: 

警告: 编译器会把上面的两个声明当成 完全不同的两个类型 ,所以是非法的。

1.3.4 匿名结构体的使用场景

       说实话,这个匿名结构体的使用次数应该要很少,他的使用场景只有在使用一次之后就不在使用,之后永远不在使用。 

1.4 结构体的自引用(错误)

1.4.1 介绍结构体自引用

       在讲结构体之前,我们来了解一下数据结构。数据结构有:线性表、栈和队列、串(KMP)、数与二叉树、图、查找、排序。(在之后的笔记中,我也会详细地写出数据结构)。

       在数据结构中,我们线性表中的链表与结构体的自引用有一定的关系。正如标题所示那样,结构体的自引用是错误的,而真正正确的是链表的写法。

错误的写法:

正确的写法:
       在链表中,我们的数据不同于数组一样是连续的放在一起,而链表是将数据不连续的放在内存空间中,我们怎么找到下一个结点呢?在内存中,每一个数据都有一个自己的地址,我们如果将下一个结点的地址存在这一个结点中,就能找到下一个结点

1.4.2 结构体自引用的一些问题

问题:

typedef struct{int a;Node* n;
}Node;
//这样写代码,可以吗?

答案:

       显然是错误的,这是一个先有鸡还是先有蛋的问题。原因在于这个结构体在创立的时候,还没有来得及给其重命名,就已经有了重命名后的指针,所以是错误的。

改进方法:

typedef struct Node {int a;struct Node* next;
}Node;

1.5 结构体变量的定义和初始化

       在标题为1.2.2中,我们讲述了如何创建结构体的全局变量和局部变量。那么接下来,我们来简单介绍一下初始化。说起初始化,想必大家都不陌生,在之前的课程中,我们就会对基本结构数据类型进行初始化。

第一种写法:

struct Point
{int x;int y;
}p1 = { 1,2 }; //声明类型的同时定义变量p1,并初始化p1

第二种写法:

//初始化:定义变量的同时赋初值。
int x = 20;
int y = 40;
struct Point p3 = { x, y };

第三种写法:

struct Stu        //类型声明
{char name[15];//名字int age;      //年龄
};
//我们可以使用下面这一种方法,不用按照顺序进行初始化
struct Stu s1 = { .age = 19, .name = "hskod" };

下面,我们来介绍一下结构体的嵌套使用

struct Node
{int data;struct Point p;struct Node* next;
}n1 = { 10, {4,5}, NULL }; //结构体嵌套初始化
struct Node n2 = { 20, {5, 6}, NULL };//结构体嵌套初始化

1.6 结构体内存对齐

1.6.1 引出offsetof

       在这之前,我们先来写一段前言用于激发读者思考!我们来看下面这两个结构体大小的对比,先来预告一波:结构体的内存对齐与结构体时候如何计算内存大小是有关。看下图:

struct Stu1 {int a;char c1;char c2;
};struct Stu2 {char c1;int a;char c2;
};

       在这两个结构体中,所有的变量都是一样的,只有变量的顺序是不一样的,这种顺序会造成结构体的内存大小是不同。为什么呢?这就和下面要介绍的内存对齐有关!

       为了便于更好地理解这一现象,我们来引出一个宏:offsetof。这个宏的作用是:此具有函数形式的宏返回数据结构或联合类型类型中成员成员的偏移值(以字节为单位)返回的值是类型为 size_t 的无符号整数值,具有指定成员与其结构开头之间的字节数

举几个例子:

1.6.2 结构体的内存对齐讲解

我们先来了解一下在C语言中结构体的内存对齐的相关规则:

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

其他成员变量要对齐到摸一个数字(对齐数)的整数倍的地址处;对齐数 == 编译器模拟的一个对齐数

1. 第一个成员在与结构体变量偏移量为 0的地址处。
2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数 = 编译器默认的一个对齐数 与 该成员大小的 较小值。
VS中默认的值为8
3. 结构体总大小为最大对齐数(每个成员变量都有一个对齐数)的整数倍。
4. 如果嵌套了结构体的情况,嵌套的结构体对齐到自己的最大对齐数的整数倍处,结构体的整
体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍。
为什么存在内存对齐 ?
大部分的参考资料都是如是说的:
1. 平台原因 ( 移植原因 )
不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特
定类型的数据,否则抛出硬件异常。
2. 性能原因
数据结构 ( 尤其是栈 ) 应该尽可能地在自然边界上对齐。
原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访
问。
总体来说:
结构体的内存对齐是拿 空间 来换取 时间 的做法。
那在设计结构体的时候,我们既要满足对齐,又要节省空间,如何做到:
让占用空间小的成员尽量集中在一起

1.7 修改默认对齐数

1.8 结构体传参


学习产出:

  1. 结构体的相关知识
  2. 结构体的内存对齐
  3. 结构体实现位段
  4. 枚举的相关知识
  5. 联合的相关知识

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

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

相关文章

Windowsold文件夹作用以及删除方法

引言 2021年6月24日,微软正式发布全新操作系统Windows 11。Windows 11系统于2021年10月5日开始全面推送。2021年10月以后生产的电脑已经预装Windows 11系统。刚开始会有一部分人不适应win 11系统,会选择退回win10。现在win11已经推出了稳定版&#xff0…

【C++】Vector -- 详解

一、vector的介绍及使用 1、vector的介绍 https://cplusplus.com/reference/vector/vector/ vector 是表示可变大小数组的序列容器。 就像数组一样,vector 也采用的连续存储空间来存储元素。也就是意味着可以采用下标对 vector 的元素进行访问,和数组一…

JS使用setInterval导致堆溢出

问题描述 使用setInterval运行较长一段时间后出现堆溢出的情况。 代码类似于 setInterval(sendHeartbeat, 30000);function sendHeartbeat() {axios.get(url).then(res > {console.log("success")}).catch(err > {console.error(err.message);}) }在一些老…

golang gin框架1——简单案例以及api版本控制

gin框架 gin是golang的一个后台WEB框架 简单案例 package mainimport ("github.com/gin-gonic/gin""net/http" )func main() {r : gin.Default()r.GET("/ping", func(c *gin.Context) {//以json形式输出,还可以xml protobufc.JSON…

IPT2602协议-USB 快速充电端口控制器

产品描述: IPT2602是一款USB端口快速充电协议控制芯片。IPT2602智能识别多种快速充电协议,对手机等受电设备进行快速充电。IPT2602根据受电设备发送的电压请求能够精确的调整VBUS输出电压,从而实现快速充电。 IPT2602在调整5V输出电压前会自动…

关掉在vscode使用copilot时的提示音

1. 按照图示的操作File --> Preferences --> Settings 2. 搜索框输入关键字Sound,因为是要关掉声音,所以找有关声音的设置 3. 找到如下图所示的选项 Audio Cues:Line Has Inline Suggetion,将其设置为Off 这样,就可以关掉suggest code时…

Vue中如何进行分布式搜索与全文搜索(如Elasticsearch)

在Vue中实现分布式搜索与全文搜索(使用Elasticsearch) 分布式搜索和全文搜索在现代应用程序中变得越来越重要,因为它们可以帮助用户快速查找和检索大量数据。Elasticsearch是一种强大的分布式搜索引擎,它可以用于实现高性能的全文…

pytorch的pixel_shuffle转tflite文件

torch.pixel_shuffle()是pytorch里面上采样比较常用的方法,但是和tensoflow的depth_to_space不是完全一样的,虽然看起来功能很像,但是细微是有差异的 def tf_pixelshuffle(input, upscale_factor):temp []depth upscale_factor *upscale_f…

ROS2 库包设置和使用 Catch2 进行单元测试

说明 本文的目的是了解如何在 ROS2 中创建库,以供其他 ROS2 包使用。除此之外,本文还介绍了如何使用 catch2 框架编写单元测试。本文的第 1 部分将详细介绍如何创建库包。第 2 部分将介绍 ROS2 软件包如何利用创建的库 上篇 ROS2 库包设置和使用 Catch2…

操作系统知识

操作系统基础 什么是操作系统? 通过以下四点可以概括操作系统到底是什么: 操作系统(Operating System,简称 OS)是管理计算机硬件与软件资源的程序,是计算机的基石。操作系统本质上是一个运行在计算机上的…

矢量图形编辑软件illustrator 2023 mac特点介绍

illustrator 2023 mac是一款矢量图形编辑软件,用于创建和编辑排版、图标、标志、插图和其他类型的矢量图形。 illustrator mac软件特点 矢量图形:illustrator创建的图形是矢量图形,可以无限放大而不失真,这与像素图形编辑软件&am…

基于虚拟阻抗的下垂控制——孤岛双机并联Simulink仿真

💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…