C语言-结构体-详解

news/2024/10/4 9:58:19/文章来源:https://blog.csdn.net/2401_86587256/article/details/142274248

博客主页:【夜泉_ly】
本文专栏:【C语言】
欢迎点赞👍收藏⭐关注❤️

C语言-结构体-详解

  • 1.前言
  • 2.结构体类型
    • 2.1声明
    • 2.2变量的创建与初始化
    • 2.3访问
    • 2.4匿名结构体类型
  • 3.结构体内存对齐
    • 3.1对齐规则
    • 3.2示例

1.前言

在C语言中,除了整型、浮点型等给定的类型外,还有很多自定义类型,结构体就是其中之一。
结构体十分重要,想要学好数据结构,必须掌握指针、结构体和动态内存管理。
本篇,我将详细介绍C语言中的结构体。

2.结构体类型

结构体作为C语言的一种重要的数据类型,其特点是由一组数据组合而成,且这些数据的类型可以不同。

2.1声明

基本框架:

struct tag
{member-list;
}variable-list;

其中,tag为该结构体的名字,member-list为成员列表,variable-list为创建变量的列表。
例如:

struct student
{char name[20];int age;int tele;
}s1,s2;

为了存储一个学生的信息,需要存储姓名、年龄、电话等等内容,这些内容的数据类型显然不同,因此,可以用结构体存储:

struct student

这里定义了一个结构体类型struct student,代表该结构体类型用于存储学生信息。
在这之后:

{char name[20];int age;int tele;
}

这里的namenametele称为结构体的成员,共同组成成员列表。
最后:

s1,s2;

这里使用结构体类型创建了两个变量s1s2,代表学生一和学生二。
当然,这里也可以不创建变量,但需注意, 分号一定不能丢 !

顺带一提,好像只要顺序正确,都能编译成功:
在这里插入图片描述
不过这样的代码过于恶心,这里只是尝试一下,平时可不敢这样写😂。

2.2变量的创建与初始化

创建:

struct Stu
{char name[20];int age;int tele;
}s1;
struct Stu S2;
int main()
{struct Stu S3;return 0;
}

S1、S2为全局变量,S3为局部变量


初始化大致分两种:

  • 按原顺序进行初始化
struct Stu s1 = { "张三", 18, 11223344 };
  • 按指定顺序进行初始化
struct Stu s2 = { .age = 81, .tele = 44332211, .name = "李四" };

2.3访问

直接访问
使用.

struct Stu s1 = { "张三", 18, 11223344 };
printf("Name: %s\nAge: %d\nTele: %d\n", s1.name, s1.age, s1.tele);

运行结果如下:
在这里插入图片描述

通过指针访问
常见于数据结构部分,因为这部分的结构体变量大多是在堆上分配的。

struct Stu *p = malloc(sizeof(struct Stu));
if(!p){perror("malloc");return 1;}
strcpy(p->name, "李四");
p->age = 81;
p->tele = 44332211;printf("Name: %s\n", p->name);
printf("Age: %d\n", p->age);
printf("Tele: %s\n", p->tele);free(p);
p = NULL;

运行结果如下:
在这里插入图片描述

一个小细节:

p->name = "李四";
//strcpy(p->name, "李四");

为什么不能写成这样?
在C语言中,字符串常量(如 “李四”)是不可修改的常量数组。
当试图将一个字符串常量赋值给一个字符数组时,编译器会报错:
在这里插入图片描述
因此,需要先创建一个字符数组,然后将字符串strcpy复制到这个数组中。

2.4匿名结构体类型

特点是没有名字,且只能用一次:

struct
{char name[20];int age;int tele;
}s,*ps;

使用时,只能在结构体后创建变量,如s*ps
且,即便两个匿名结构体的成员列表相同,它们依然是两个不同的类型。

3.结构体内存对齐

先来看看下面这段代码:

struct S1
{char c1;int n;char c2;
};
struct S2
{char c1;char c2;int n;
};
int main()
{printf("%zd\n",sizeof(struct S1));printf("%zd\n",sizeof(struct S2));return 0;
}

运行结果如下:

12
8

如果单纯的将成员列表中各个成员的大小相加,那么S1、S2都应该是6字节。
但是很遗憾,不仅不是6字节,S1、S2的大小甚至都不一样,这是为什么呢?

结构体的大小不是结构体元素单纯相加就行的,因为主流的计算机使用的都是32bit字长的CPU,对这类型的CPU取4个字节的数要比取一个字节要高效,也更方便。所以在结构体中每个成员的首地址都是4的整数倍的话,取数据元素时就会相对更高效,这就是内存对齐的由来。

3.1对齐规则

  1. 第一个成员对齐到偏移量为0的位置。
  2. 之后的成员对齐,对齐到对齐数的整数倍处。
  3. 总大小需是最大对齐数的整数倍。

对齐数

  • 常见类型的对齐数,按默认对齐数和自身长度,较小的那个进行
  • 数组的对齐数,按默认对齐数和自身成员长度,较小的那个进行
  • 结构体的对齐数,按默认对齐数和自身成员最大长度,较小的那个进行

默认对齐数
每个特定平台上的编译器都有自己的默认对齐数,如VS上的是8。
也可以通过预编译指令修改默认对齐数:

#pragma pack(n)

注:这里的n最好取2的倍数

3.2示例

下面我将给出多个结构体,并解释它们的大小是如何得到的。


struct S1
{char c1;char c2;int n;
}

先将c1存入偏移量为0的位置,占一字节:
在这里插入图片描述
再存c2,计算得对齐数为1,因此存入偏移量为1的位置,占一字节:
在这里插入图片描述
再存n,计算得对齐数为4,因此存入偏移量为4的位置,占四字节:
在这里插入图片描述
最后,最大对齐数为4,因此大小为8


struct S2
{char c1;int n;char c2;
};

先将c1存入偏移量为0的位置,占一字节:
在这里插入图片描述
再存n,计算得对齐数为4,因此存入偏移量为4的位置,占四字节:
在这里插入图片描述
再存c2,计算得对齐数为1,因此存入偏移量为8的位置,占一字节:
在这里插入图片描述
最后,最大对齐数为4,因此大小为12


struct S3
{short n;char c;int i;
};

先将n存入偏移量为0的位置,占两字节:
在这里插入图片描述

再存c,计算得对齐数为1,因此存入偏移量为2的位置,占一字节:
在这里插入图片描述

再存i,计算得对齐数为4,因此存入偏移量为4的位置,占四字节:

在这里插入图片描述

最后,最大对齐数为4,因此大小为8


struct S4
{char c;struct S3 s3;double d;
};

先将c存入偏移量为0的位置,占一字节:
在这里插入图片描述
再存S3,计算得对齐数为4,因此存入偏移量为4的位置,占八字节:
在这里插入图片描述
再存d,计算得对齐数为8,因此存入偏移量为16的位置,占八字节:
在这里插入图片描述
最后,最大对齐数为8,因此大小为24


希望本篇文章对你有所帮助!并激发你进一步探索C语言的兴趣!

本人仅是个C语言初学者,如果你有任何疑问或建议,欢迎随时留言讨论!让我们一起学习,共同进步!

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

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

相关文章

开源免费的工贸一体行业ERP管理系统

引言 在当今数字化浪潮汹涌澎湃的时代,中小企业面临着前所未有的挑战与机遇。如何实现数字化转型发展,成为了众多中小企业主心头的大事。 据相关数据显示,目前我国中小企业数量已经超过了 4000 万户,然而成功实现数字化转型的比例…

C语言-数据结构 有向图拓扑排序TopologicalSort(邻接表存储)

拓扑排序算法的实现还是比较简单的,我们需要用到一个顺序栈辅助,采用邻接表进行存储,顶点结点存储入度、顶点信息、指向邻接结点的指针,算法过程是:我们先将入度为0的顶点入栈,然后弹出栈顶结点&#xff0c…

【Linux系统编程】用互斥量和信号量加锁STL容器,避免并发问题

目录 引言 容器模型 容器代码 个人主页:东洛的克莱斯韦克-CSDN博客 引言 STL容器并没有保证线程安全,而大多数应用场景下,为了追求效率,多线程是必不可少的。而底层容器难免会有并发问题。从设计上来说要么在上层代码做加锁处…

C Primer Plus 第5章习题

你该逆袭了 红色标注的是:错误的答案 蓝色标注的是:正确的答案 绿色标注的是:做题时有疑问的地方 橙色标注的是:答案中需要着重注意的地方 练习题 一、复习题1、2、3、4、错误答案:正确答案: 5、我的答案&a…

Ubuntu24.04部署docker

1、更新软件 apt update 2、安装curl apt install apt-transport-https curl 3、导入阿里云GPG秘钥 curl -fsSL https://mirrors.aliyun.com/docker-ce/linux/ubuntu/gpg | sudo gpg --dearmor -o /etc/apt/keyrings/docker.gpg 4、添加Docker阿里云仓库到Ubuntu 24.04的…

【Elasticsearch系列八】高阶使用

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

CPU 和 GPU:为什么GPU更适合深度学习?

目录 什么是 CPU ? 什么是 GPU ? GPU vs CPU 差异性对比分析 GPU 是如何工作的 ? GPU 与 CPU 是如何协同工作的 ? GPU vs CPU 类型解析 GPU 应用于深度学习 什么是 CPU ? CPU(中央处理器)…

从0书写一个softmax分类 李沐pytorch实战

输出维度 在softmax 分类中 我们输出与类别一样多。 数据集有10个类别,所以网络输出维度为10。 初始化权重和偏置 torch.norma 生成一个均值为 0,标准差为0.01,一个形状为size(num_inputs, num_outputs)的张量偏置生成一个num_outputs 10 的一维张量&a…

PyQt5-QCheckBox-开关按钮

效果预览 实现代码 from PyQt5.QtWidgets import QCheckBox, QApplication, QWidget, QVBoxLayout from PyQt5.QtCore import Qt, QRect, QPropertyAnimation, QEasingCurve, pyqtProperty from PyQt5.QtGui import QPainter, QColor, QPen, QFontclass CompactSwitchCheckbox…

基于STM32的温度、电流、电压检测proteus仿真系统(OLED、DHT11、继电器、电机)

目录 一、主要功能 二、硬件资源 三、程序编程 四、实现现象 一、主要功能 基于STM32F103C8T6 采用DHT11读取温度、滑动变阻器模拟读取电流、电压。 通过OLED屏幕显示,设置电流阈值为80,电流小阈值为50,电压阈值为60,温度阈值…

VSCode C++ Tasks.json中的变量

前言 上文介绍了在VSCode中创建C项目和编译多文件的情况。本文将介绍Tasks.json中一些变量的含义; 内容 tasks.json文件 下文参考VSCode文档:Visual Studio Code 变量参考 预定义标量 ${userHome} - 用户主文件夹的路径${workspaceFolder} - 在 VS Co…

02 信念·组织力·战略 - 目标管理

1,每个企业每年都要确定一年、三年目标。大多数企业的目标管理有问题。所确定的目标常常在时间段里无法达成。 2,华为的成长曲线,32年,近乎完美的成长曲线。历年销售增长图。和目标的确立和目标的达成有密切的关联。 3,尤其是通信行业,在过往的30年,现象是,那些巨无霸,…

C# AutoResetEvent ManualResetEvent Mutex 对比

三个函数功能类似,都是线程同步的主要函数。但在使用上有一些差别。 关于代码的使用,帖子很多。形象的用图来描述一下。

算法:30.串联所有单词的子串

题目 链接:leetcode链接 思路分析(滑动窗口) 这道题目类似寻找异位词的题目,我认为是寻找异位词的升级版 传送门:寻找异位词 为什么说像呢? 注意:这道题目中words数组里面的字符串长度都是相同的&…

【高中数学/三角函数/判别式法】设x>0,y>0,x+2y=5,则(x+1)(2y+1)/(xy)^0.5的最小值为?

【问题】 设x>0,y>0,x2y5,则(x1)(2y1)/(xy)^0.5的最小值为? 【出处】 《解题卡壳怎么办--高中数学解题智慧剖析》P38页第3题首问 余继光、苏德矿著 【解答】 由 x2y5 可设x5cosθ^2,y5sinθ^2/2 代入目标式得(x1)(2y1)/(xy)^0.5(5cosθ^21)(5sinθ^21)…

【MySQL】MySQL和Workbench版本兼容问题

1、安装MySQL WorkBench 最新版本下载:https://dev.mysql.com/downloads/workbench/ 历史版本下载:https://downloads.mysql.com/archives/workbench/ 2、问题描述 本人在Windows下安装了一个旧版本的MySQL(5.1),同…

滑模控制算法——基本原理(附MATLAB程序)

滑模控制算法(Sliding Mode Control, SMC)是一种非线性控制策略,广泛用于处理系统的非线性、扰动以及建模不确定性。它的核心思想是通过设计一个滑模面,使得系统状态在滑模面上滑动,从而实现系统的稳定性和期望的动态性…

策略路由与路由策略的区别

🐣个人主页 可惜已不在 🐤这篇在这个专栏 华为_可惜已不在的博客-CSDN博客 🐥有用的话就留下一个三连吧😼 目录 一、主体不同 二、方式不同 三、规则不同 四、定义和基本概念 一、主体不同 1、路由策略:是为了改…

7 递归——206. 反转链表 ★

7 递归 206. 反转链表 给你单链表的头节点head,请你反转链表,并返回反转后的链表。 示例 1: 输入:head = [1,2,3,4,5] 输出:[5,4,3,2,1] 算法设计 可以充分利用原有的存储空间,通过修改指针实现单链表的就地逆置。相当于将所有的箭头反向,头指针指向原链表的尾部。…

《OpenCV计算机视觉》—— 身份证号码识别案例

文章目录 一、案例实现的整体思路二、代码实现1.首先定义两个函数2.模板图像中数字的定位处理3.身份证号码数字的定位处理4.使用模板匹配,计算匹配得分,找到正确结果 一、案例实现的整体思路 下面是一个数字0~9的模板图片 案例身份证如下: 对…