【C语言】结构体

目录

  • 1. 前言
  • 2. 结构体类型的声明
    • 2.1 结构体的概念
    • 2.2 结构的创建
    • 2.3 特殊的声明
    • 2.4 结构的自引用
  • 3. 结构成员访问操作符
  • 4. 结构体内存对齐
    • 4.1 对齐规则
    • 4.2 为什么存在内存对齐?
    • 4.3 修改默认对齐数
  • 5. 结构体传参
  • 6. 结构体实现位段
    • 6.1 什么是位段
    • 6.2 位段的内存分配
    • 6.3 位段的跨平台问题
    • 6.4 位段的应用
    • 6.5 位段使用的注意事项

1. 前言

在C语言中,有两种类型,一种是内置类型,可以直接使用,包括char short int long long long float double;一种是自定义类型,当内置类型不能满足时,支持自定义一些类型,像结构体、枚举、联合体。
这次先来看看结构体。

2. 结构体类型的声明

2.1 结构体的概念

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

2.2 结构的创建

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

在这里插入图片描述
这里的tag可以换成自己指定的名字,member-list可以换成多个成员变量,注意大括号外面必须有分号。

举个例子:描述一个学生

struct Stu
{char name[20];//名字int age;//年龄char sex[5];//性别float score;//成绩}; //分号不能丢

定义结构体变量:

  1. 全局变量
struct Student
{char name[20];int age;char sex[5];float score;
} s1, s2, s3;//s1, s2, s3 是三个结构体变量 - 全局变量
  1. 局部变量
struct Student
{char name[20];int age;char sex[5];float score;
}int main()
{struct Student s1, s2, s3;//局部变量return 0;
}

2.3 特殊的声明

在声明结构的时候,可以不完全的声明。
比如:

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

这种叫匿名结构体。
当我们想使用指针来给有相同成员变量的匿名结构体,赋值时,发现会失败。

struct
{char name[20];char author[12];float price;
}b={0};struct
{char name[20];char author[12];float price;
}* p;int main()
{p = &b;//不建议这样写return 0;
}

在这里插入图片描述
在编译器来看它们是不相同的指针类型。

注意:

  1. 编译器会把上面的两个声明当成完全不同的两个类型,所以是非法的。
  2. 匿名的结构体类型,如果没有对结构体类型重命名的话,基本上只能使用一次。

2.4 结构的自引用

在结构中包含⼀个类型为该结构本⾝的成员是否可以呢?
比如,定义⼀个链表的节点:

struct Node
{int data;//数据struct Node next;//下一个节点
};
int main()
{struct Node n;return 0;
}

那么这里的n占多少个字节?
这里一个结构体里面有int占4个字节,struct Node next存放下一个节点,里面有int占4个字节,一直重复,就不能算出。
就比如在一个房子子里放同样大小的房子,是放不进去的。

但是使用结构体指针就可以解决这个问题。

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

在结构体自引用使用的过程中,夹杂typedef对匿名结构体类型重命名,也容易引入问题,看看下面的代码,可行吗?

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

答案是不行的,因为Node是对前面的匿名结构体类型的重命名产生的,但是在匿名结构体内部提前使用Node类型来创建成员变量,这是不行的。
所以定义结构体不要使用匿名结构体了。

3. 结构成员访问操作符

结构成员访问操作符有两个⼀个是 . ,⼀个是 -> .
形式如下:

  1. 结构体变量.成员变量名
  2. 结构体指针—>成员变量名
    举个例子:
#include <stdio.h>
#include <string.h>
struct Stu
{char name[15];//名字int age; //年龄
};
void print_stu(struct Stu s)
{printf("%s %d\n", s.name, s.age);
}
void set_stu(struct Stu* ps)
{strcpy(ps->name, "李四");ps->age = 28;
}
int main()
{struct Stu s = { "张三", 20 };print_stu(s);set_stu(&s);print_stu(s);return 0;
}

在这里插入图片描述

4. 结构体内存对齐

怎么计算结构体的大小呢?
这里需要先了解结构体内存对齐

4.1 对齐规则

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

  1. 结构体的第⼀个成员对齐到相对结构体变量起始位置偏移量为0的地址处
  2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
    对齐数 = 编译器默认的⼀个对齐数与该成员变量大小的较小值
    VS中默认的值为8
    Linux中没有默认对齐数,对齐数就是成员自身的大小
  3. 结构体总大小为最大对齐数(结构体中每个成员变量都有一个对齐数,所有对齐数中最大的)的整数倍。
  4. 如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。
    来看看这个例子:

struct S1
{char c1;char c2;int a;
};struct S2
{char c1;int a;char c2;
};int main()
{struct S2 s2 = { 'a', 100, 'b'};printf("%zd\n", sizeof(s2));struct S1 s1 = { 'a', 'b', 100 };printf("%zd\n", sizeof(s1));return 0;
}

在这里插入图片描述
这两个结构体里面的成员变量明明都一样,为什么它们结构体大小确不相同呢?
对于s1而言:char c1,占一个字节,而VS中默认的值为8,1小,所以选择1,而结构体的第⼀个成员对齐到相对结构体变量起始位置偏移量为0的地址处。所以c1就占了0。
char c2,占一个字节,而VS中默认的值为8,1小,所以选择1;因为其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。这里1刚好就是1的整数倍。
int a,占4个字节,而VS中默认的值为8,4小,所以选择4;因为其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处,4的整数倍就是4,然后从4开始占4个内存空间。
总的用了8个地址空间

最后最后因为结构体总大小为最大对齐数(结构体中每个成员变量都有一个对齐数,所有对齐数中最大的)的整数倍,这里最大的为4,所以就是8

在这里插入图片描述
对于s2而言:
char c1和s1中的一样。
int a,占4个字节,而VS中默认的值为8,4小,所以选择4;4的整数倍就是4,然后从4开始占4个内存空间。
char c2,占一个字节,而VS中默认的值为8,1小,所以选择1;这里8刚好就是1的整数倍。
总的共用了9个
最后因为结构体总大小为最大对齐数(结构体中每个成员变量都有一个对齐数,所有对齐数中最大的)的整数倍,这里最大的为4,所以最后结构体的大小就是12
在这里插入图片描述
对于嵌套

struct S3
{double d;char c;int i;
};struct S4
{char c1;struct S3 s3;double d;
};int main()
{printf("%zd\n", sizeof(struct S3));printf("%zd\n", sizeof(struct S4));return 0;
}

在这里插入图片描述
在这里插入图片描述
对于s3而言:很简单,像上面两个那种就行。

但对于s4而言:它嵌套了s3,它有一个规定 :如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。所以这里的s3就从8开始,然后占16个地址空间,到23, double d,占8个字节,而VS中默认的值为8,所以选择4,就从24开始。
一共32,恰好是8的倍数。所以最后结果就是32。

4.2 为什么存在内存对齐?

大部分的参考资料都是这样说的:
1. 平台原因(移植原因):
不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
2. 性能原因:
数据结构(尤其是栈)应该尽可能地在自然然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。假设一个处理器总是从内存中取8个字节,则地址必须是8的倍数。如果我们能保证将所有的double类型的数据的地址都对齐成8的倍数,那么就可以用一个内存操作来读或者写值了。否则,我们可能需要执行两次内存访问,因为对象可能被分放在两个8字节内存块中。
总体来说:结构体的内存对齐是拿空间来换取时间的做法。
在这里插入图片描述
这里如果不存在内存对齐,当在内存中拿数据时,如果一次性取的是4个字节,在第一个图中那么就要读两次。
当浪费了一些空间,对齐时,一次读4个字节时,对a的读取,一次就可以了。

4.3 修改默认对齐数

#pragma 这个预处理指令,可以改变编译器的默认对齐数。
#pragma pack()取消设置的默认对齐数,还原为默认。
举个例子:

#include <stdio.h>#pragma pack(1)//设置默认对齐数为1
struct S
{char c1;int i;char c2;
};
#pragma pack()//取消设置的默认对⻬数,还原为默认
int main()
{printf("%zd\n", sizeof(struct S));return 0;

在这里插入图片描述

5. 结构体传参

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

  1. 函数传参的时候,参数是需要压栈,会有时间和空间上的系统开销。
  2. 如果传递一个结构体对象的时候,结构体过⼤,参数压栈的的系统开销比较大,所以会导致性能的下降。
    看一个示例:
struct S
{int data[1000];int num;
};void print1(struct S t)
{int i = 0;for (i = 0; i < 10; i++){printf("%d ", t.data[i]);}printf("\n");printf("num = %d\n", t.num);
}void print2(const struct S* ps)
{int i = 0;for (i = 0; i < 10; i++){printf("%d ", ps->data[i]);}printf("\n");printf("num = %d\n", ps->num);
}int main()
{struct S s = { {1,2,3,4}, 1000 };print1(s);//传递结构体变量 - 传值调用print2(&s);//传递结构体变量的地址 - 传址调用return 0;
}

在这里插入图片描述
这里⾸选print2函数。

6. 结构体实现位段

结构体讲完就得讲讲结构体实现 位段 的能力

6.1 什么是位段

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

  1. 位段的成员必须是 int、unsigned int 或signed int ,在C99中位段成员的类型也可以
    选择其他类型。
  2. 位段的成员名后边有⼀个冒号和⼀个数字。
    比如
struct A
{int _a:2;int _b:5;int _c:10;int _d:30;
};

举个例子:

struct A
{int _a : 2;//只占2个二进制位int _b : 5;//只占5个二进制位int _c : 10;//只占10个二进制位int _d : 30;//只占30个二进制位
};struct B
{int _a;int _b;int _c;int _d;
};int main()
{printf("%d\n", sizeof(struct A));printf("%d\n", sizeof(struct B));return 0;
}

在这里插入图片描述

那么A的大小是怎么计算的呢?

6.2 位段的内存分配

  1. 位段的成员可以是 int unsigned int signed int 或者是 char 等类型
  2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。
  3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。

struct S
{char a : 3;char b : 4;char c : 5;char d : 4;
};int main()
{struct S s = { 0 };s.a = 10;s.b = 12;s.c = 3;s.d = 4;printf("%d\n", sizeof(struct S));return 0;
}

在这里插入图片描述
当开辟了内存后,内存中的每个比特位从右向左使用。
在前面使用后,剩余的空间不足下一个成员使用的时候,剩余空间就不用了。
所以内存中应该是下面这样的存储。
在这里插入图片描述
发现果然是这样

在这里插入图片描述

6.3 位段的跨平台问题

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

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

6.4 位段的应用

下图是网络协议中,IP数据报的格式,我们可以看到其中很多的属性只需要几个bit位就能描述,这里使用位段,能够实现想要的效果,也节省了空间,这样网络传输的数据报大小也会较小一些,对网络的畅通是有帮助的。
在这里插入图片描述

6.5 位段使用的注意事项

位段的几个成员共有同⼀个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置处是没有地址的。内存中每个字节分配⼀个地址,⼀个字节内部的bit位是没有地址的。
所以不能对位段的成员使用&操作符,这样就不能使用scanf直接给位段的成员输入值,只能是先输入放在⼀个变量中,然后赋值给位段的成员

有问题请指出,大家一起进步!

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

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

相关文章

[ACTF2020 新生赛]BackupFile

打开题目就一句话&#xff1a;尝试找到源文件 和上一题一样&#xff0c;用dirsearch扫描网站找到了一下内容 flag.php&#xff0c;0B&#xff0c;虚假flag 瞅一眼index.php.bak是啥 下载了一个文件&#xff0c;把bak后缀删掉&#xff0c;打开了index.php源码 is_numeric()&am…

成为AI产品经理——模型评估(混淆矩阵)

一、混淆矩阵 1.混淆矩阵的介绍 混淆矩阵有两个定义positive&#xff08;正例&#xff09;和negative&#xff08;反例&#xff09;。分别代表模型结果的好和坏。 下图就是一个分类问题的混淆矩阵。横行代表真实的情况&#xff0c;而竖行代表预测的结果。 为了便于理解&…

新疆大学与优艾智合机器人成立联合创新实验室

11月22日至24日&#xff0c;第五届中国工业互联网大赛新疆赛站决赛在新疆维吾尔自治区昌吉回族自治州昌吉市举行。在大赛中崭露头角的优秀解决方案&#xff0c;将为绿色工厂、绿色园区、绿色供应链等建设提供新的动能&#xff0c;促进工业绿色发展。 作为大赛的成果延伸&#…

面试常问-如何判断链表有环、?

如何判断链表有环 题目&#xff1a;解决方案一&#xff1a;解决方案二&#xff1a;解决方案三&#xff1a; 题目&#xff1a; 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;…

MSB3541 Files 的值“<<<<<<< HEAD”无效。路径中具有非法字符。

MSB3541 Files 的值“<<<<<<< HEAD”无效。路径中具有非法字符。 一般来说出现这个问题是因为使用git版本控制工具合并代码出现了问题&#xff0c;想要解决也很简单。 如图点击错误后定位到文件&#xff0c;发现也没有什么问题。 根据错误后边的提示&a…

JVM内存模型和结构详解(五大模型图解)

目录 方法区&#xff08;Method Area&#xff09;: 堆&#xff08;Heap&#xff09;: 栈&#xff08;Stack&#xff09;: 本地方法栈&#xff08;Native Method Stack&#xff09;: 程序计数器&#xff08;Program Counter Register&#xff09;: Java Virtual Machine (J…

SpringCloudAlibaba之sentinel 流量卫兵(流控,熔断降级) ——详细讲解

目录 一、什么是sentinel 二、sentinel使用 1. sentinel dashboard的安装 2.启动 3.访问web界面 ​编辑 4.登录 三、sentinel 实时监控服务 1.创建项目引入依赖 2.配置 3.启动服务 4.访问dashboard界面查看服务监控 5.开发服务 6.启动进行调用 7.查看监控界面 四、senti…

肠道菌群16s检测粪便采样工具包 粪便采样套装

肠道菌群16s检测是一种常见的分子生物学技术&#xff0c;用于研究人体肠道中的微生物群落。该技术通过分析16s rRNA基因序列&#xff0c;可以快速、准确地鉴定并定量不同种类的肠道微生物。 肠道菌群16s检测通常通过采集粪便样本进行分析。在实验室中&#xff0c;通过提取微生物…

leetcode面试经典150题——33 最小覆盖子串(滑动窗口)

题目&#xff1a; 最小覆盖子串 描述&#xff1a; 给你一个字符串 s 、一个字符串 t 。返回 s 中涵盖 t 所有字符的最小子串。如果 s 中不存在涵盖 t 所有字符的子串&#xff0c;则返回空字符串 “” 。 注意&#xff1a; 对于 t 中重复字符&#xff0c;我们寻找的子字符串中…

【三维重建】摄像机标定(张正友相机标定法)

摄像机标定的目的是为了求解摄像机的内、外参数 求解投影矩阵M 通过建立特殊的场景&#xff0c;我们能过得到多对世界坐标和对应图像坐标 根据摄像机几何可知 &#xff1a; &#xff0c;M是一个3*4的矩阵&#xff0c;令 通过一对点可以得到两个方程组&#xff0c;M中一共有11个…

SpringBoot : ch09 整合Redis

前言 当你的应用程序需要一个快速、可扩展的内存数据库时&#xff0c;Redis是一个非常流行的选择。通过将Redis与Spring Boot集成&#xff0c;你可以轻松地利用Redis的功能&#xff0c;例如缓存、会话存储和消息队列等&#xff0c;从而提升应用程序的性能和可伸缩性。 在本教…

mongodb查询数据库集合的基础命令

基础命令 启动mongo服务 mongod -f /usr/local/mongodb/mongod.conf //注意配置文件路径停止mongo服务 关闭mongodb有三种方式&#xff1a; 一种是进入mongo后通过mongo的函数关闭&#xff1b; use admin db.shutdownServer()一种是通过mongod关闭&#xff1b; mongod --s…

Selenium 学习(0.14)——软件测试之测试用例设计方法——因果图法2【基本步骤及案例】

1、因果图法的基本步骤 2、案例分析 1&#xff09;分析原因和结果 2&#xff09;关联原因和结果 投入1元5角或2元&#xff0c;按下“可乐”&#xff0c;送出“可乐”【暂时忽略找零】 投入2元&#xff0c;按下“可乐”或“雪碧”。找零5角&#xff0c;送出“可乐”或“雪…

软件测试测试文档编写

在软件测试中的流程中&#xff0c;测试文档也是一个重要的流程&#xff0c;所以测试人员也需要学习测试文档的编写和阅读。 一、定义&#xff1a;   测试文档&#xff08;Testing Documentation&#xff09;记录和描述了整个测试流程&#xff0c;它是整个测试活动中非常重要…

vscode注释插件「koroFileHeader」

前言 在vscode上进行前端开发&#xff0c;有几个流行的注释插件&#xff1a; Better CommentsTodo TreekoroFileHeaderDocument ThisAuto Comment Blocks 在上面的插件中我选择 koroFileHeader 做推荐&#xff0c;原因一是使用人数比较多&#xff08;最多的是 Better Commen…

NAS-DIP: Learning Deep Image Prior with Neural Architecture Search

NAS-DIP: 用神经结构搜索学习深度图像先验 论文链接&#xff1a;https://arxiv.org/abs/2008.11713 项目链接&#xff1a;https://github.com/YunChunChen/NAS-DIP-pytorch Abstract 最近的研究表明&#xff0c;深度卷积神经网络的结构可以用作解决各种逆图像恢复任务的结构…

使用vue脚手架创建vue项目

Vue是一个流行的前端框架&#xff0c;可以用简洁的语法和组件化的思想开发单页面应用。Vue脚手架是一个官方提供的命令行工具&#xff0c;它可以帮你快速搭建和配置vue项目的基本结构和依赖。 本文介绍如何使用vue脚手架创建一个vue2项目&#xff0c;并选择一些常用的功能和插件…

Java开源ETL工具-Kettle

一、背景 公司有个基于Kettle二次开发产品主要定位是做一些数据ETL的工作, 所谓的ETL就是针对数据进行抽取、转换以及加载的过程&#xff0c;说白了就是怎么对原始数据进行清洗&#xff0c;最后拿到我们需要的、符合规范的、有价值的数据进行存储或者分析的过程。 一般处理ETL的…

【从浅识到熟知Linux】基本指令之man、uname和bc

&#x1f388;归属专栏&#xff1a;从浅学到熟知Linux &#x1f697;个人主页&#xff1a;Jammingpro &#x1f41f;每日一句&#xff1a;干完饭写篇博客放松一下。 文章前言&#xff1a;本文介绍man、uname和bc指令用法并给出示例和截图。 文章目录 man基本语法功能选项无选项…

人工智能入门,什么是AlphaGo式搜索?

AlphaGo式搜索是一种搜索算法&#xff0c;它是由DeepMind开发的AlphaGo团队在开发AlphaGo程序时使用的搜索策略。 AlphaGo是一个基于人工智能的围棋程序&#xff0c;它在2016年击败了世界冠军柯洁&#xff0c;引起了广泛的关注。 AlphaGo式搜索的核心思想是使用蒙特卡洛树搜索…