C语言之自定义类型_结构体篇(1)

目录

什么是结构?

结构体类型的声明

常规声明

特殊声明-匿名结构体

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

定义

初始化

访问

嵌套结构体 

结构体的自引用

什么是结构体的自引用 

NO1.

NO2.

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

产生内存对齐

NO1 

NO2 

NO3 

内存对齐-结构体类型内存中存储

NO1

NO2

NO3

为什么要对齐

优化结构体成员顺序 

修改默认对齐数

百度笔试题


今天来深入结构体,爬了武功山很是艰辛哈哈。

C语言有内置类型:char short int long longlong float double 。但是我们生活中有负责对象需要去描述,例如人需要名字+年龄+身高等等;书需要书名+作者+出版社等等。所以C语言就有了自定义类型:结构体 枚举 联合体。今天我们重点讲解结构体!

可以先回顾一下结构体的基础知识:C语言之结构体篇_唐唐思的博客-CSDN博客

什么是结构?

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

可以与数组相比较,数组:一组相同类型元素的集合。 

结构体类型的声明

struct tag
{member-list;
}variable-list;
typedef struct 
{member-list;
}tag;
  • struct 是结构体关键字 不能省略
  • tag 是结构体名字 自己命名即可
  • member-list 是成员列表
  • variable-list 是结构体类型的变量列表
  • 分号 ; 一定不要忘记
  • 记住结构体是结构体类型,是一种变量类型 

常规声明

描述一个学生信息 

#include<stdio.h>
struct student
{char name[20];int age;char sex[5];//一个汉字占2个字节+一个\0=5个字节char id[20];//学号
}s1, s2, s3;//分号不能丢
//s1,s2,s3是三个结构体变量,全局变量
int main()
{struct student s5, s6, s7;//ss4,s5,s6是三个结构体变量,局部变量return 0;
}

描述一本书的信息 

struct book
{char name[20];char author[12];float printf;
};
  • 结构体类型声明
  • 结构体类型创建变量 

特殊声明-匿名结构体

  • 匿名结构体类型是一种特殊结构体类型 ,只能特殊声明,只能声明一次。
struct
{char name[20];char author[12];float printf;
}b1,b2;
//只能使用一次,也就是b1
//当然如果你想要创建多个结构体类型变量也是可以的

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

那么问题来了,可以使用下面这种写法吗??不建议使用哦

struct
{char name[20];char author[12];float printf;
}b1;struct
{char name[20];char author[12];float printf;
}*p;
int main()
{p = &b1;//不建议这样写,编译器会认为两端的结构体类型不一样
}

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

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

定义

#include<stdio.h>
struct student
{char name[20];int age;
}s1,s2;//定义全局变量struct student s3, s4;//定义全局变量
int main()
{struct student s5, s6;//定义局部变量
}

初始化

#include<stdio.h>
struct student
{char name[20];int age;
}s1 = { "zhangsan",20 };struct student s2 = { "lisi",25 };
int main()
{struct student s3 = { "ruhua",18 };//正序初始化struct student s3 = { .age=18,.name="ruhua"};//乱序初始化
}

访问

#include<stdio.h>
struct student
{char name[20];int age;
};
int main()
{struct student s3 = { "ruhua",18 };//正序初始化struct student s4 = { .age=18,.name="ruhua"};//乱序初始化struct student* s = &s3;printf("%d %s\n", s3.age, s3.name);printf("%d %s\n", (*s).age, (*s).name);printf("%d %s\n", s->age, s->name);
}

嵌套结构体 

#include<stdio.h>
struct Point
{int x;int y;
}p1; //声明类型的同时定义变量p1
struct Point p2; //定义结构体变量p2
struct Point p3 = {1,2};//初始化:定义变量的同时赋初值。struct Stu//类型声明
{char name[15];//名字int age;//年龄
};
struct Stu s = { "zhangsan", 20 };//初始化
struct Node
{int data;struct Point p;struct Node* next;
}n1 = { 10, {4,5}, NULL }; //结构体嵌套初始化
struct Node n2 = { 20, {5, 6}, NULL };//结构体嵌套初始化
//嵌套结构体的大小问题
#include<stdio.h>
#include<stddef.h>
struct S3
{double d;char c;int i;
};
struct S4
{char c1;struct S3 s3;double d;
};
int main()
{printf("%d\n", sizeof(struct S4));return 0;
}

结构体的自引用

什么是结构体的自引用 

 大家应该都听说过数据结构,数据结构就是数据在内存中的存储和组织结构。

 这里简单谈一下:假设要将1,2,3,4,5存储在内存中。我们会有怎样的数据结构。

线性数据结构树形数据结构

在线性数据结构中,像1这样一个数据叫 节点 ,如果我们想用结构体去表示一个节点,需要包含哪些信息呢?信息:1.节点本身的信息_数据域  2.找到下一个节点的信息——指针域

那找到下一个节点信息的关键点就是:指针。 知道我们知道下一个节点的地址,并且放入上一个节点的结构体成员 指针变量中,我们就可以轻松联系节点与节点之间的桥梁。

struct Node
{int data;//本节点信息——数据域struct Node* n;//下一个节点结构体类型的指针变量——指针域
};
typedef struct Node
{int data;//本节点信息——数据域struct Node* n;//下一个节点结构体类型的指针变量——指针域
}Node;

NO1.

问题来了,可以用匿名结构体吗?当然不可以

struct 
{int data;//本节点信息——数据域struct Node* n;//下一个节点结构体类型的指针变量——指针域
};//❌

NO2.

那下面这种写法呢? 

typedef struct 
{int data;//本节点信息——数据域Node* n;//下一个节点结构体类型的指针变量——指针域
}Node;//❌

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

我们已经掌握了结构体的基本使用了。现在我们是深入讨论一个问题:计算结构体的大小。

这也是一个特别热门的考点:结构体内存对齐。 

产生内存对齐

我们先来看端代码:

#include<stdio.h>
struct S1
{char c1;//1int i;//4char c2;//1
};//6
struct S2
{char c1;//1char c2;//1int i;//4
};//6
int main()
{printf("%d\n", sizeof(struct S1));printf("%d\n", sizeof(struct S2));return 0;
}

为什么不是按照我们预期的内存大小呢?? 我们来测试一下每个数据在内存中的偏移量。 

偏移量】偏移量_百度百科 (baidu.com) 通俗来讲 就是与起始地址(首地址)的偏移距离

offsetof是宏可以直接使用,用于计算结构体成员相较于起始位置的偏移量

头文件#include<stddef.h>,返回值是偏移量

【宏offsetof】:offsetof - C++ Reference (cplusplus.com)

】我们在后面会讲解。 大家可以现在网上了解一下宏,戳一戳:宏(计算机术语)_百度百科 (baidu.com)

NO1 

#include<stdio.h>
#include<stddef.h>
struct S1
{char c1;//1int i;//4char c2;//1
};//6
int main()
{printf("%d\n", offsetof(struct S1, c1));printf("%d\n", offsetof(struct S1, i));printf("%d\n", offsetof(struct S1, c2));return 0;
}

 

我们发现有部分空间是被浪费了的??那S2也是这样吗?我们来看看

NO2 

#include<stdio.h>
#include<stddef.h>
struct S2
{char c1;//1char c2;//1int i;//4
};//6
int main()
{printf("%d\n", offsetof(struct S2,c1));printf("%d\n", offsetof(struct S2,c2));printf("%d\n", offsetof(struct S2, i));return 0;
}

NO3 

除了上面的问题我们还有一个嵌套结构体大小的问题哟!🆗🆗

#include<stdio.h>
#include<stddef.h>
struct S3
{double d;char c;int i;
};
struct S4
{char c1;struct S3 s3;double d;
};
int main()
{printf("%d\n", sizeof(struct S4));return 0;
}

关于数据在内存中的存储,偏移量有什么存储规则吗?当然,对齐规则。  

内存对齐-结构体类型内存中存储

对齐规则也就是结构体在内存中如何存储

考虑如何计算?那我们首先要掌握结构体的对齐规则:

  1. 第一个成员在与结构体变量偏移量为0的地址处。
  2. 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
  3. 结构体总大小为最大对齐数(每个成员变量的对齐数 比较之后 最大的)的整数倍
  4. 如果嵌套了结构体的情况,嵌套的结构体 对齐到 自己的成员中对齐数 最大对齐数整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体的对齐数)的整数倍
  • 对齐数 = 编译器默认的 一个对齐数 与 该成员本省的大小 比较之后的 较小值。
  • VS中默认值为8
  • Linux中gcc没有默认对齐数,对齐数就是成员自身的大小

NO1

#include<stdio.h>
#include<stddef.h>
struct S1
{char c1;//1int i;//4char c2;//1
};//6
int main()
{printf("%d\n", offsetof(struct S1, c1));printf("%d\n", offsetof(struct S1, i));printf("%d\n", offsetof(struct S1, c2));return 0;
}

 

NO2

#include<stdio.h>
#include<stddef.h>
struct S2
{char c1;//1char c2;//1int i;//4
};//6
int main()
{printf("%d\n", offsetof(struct S2,c1));printf("%d\n", offsetof(struct S2,c2));printf("%d\n", offsetof(struct S2, i));return 0;
}

 

NO3

#include<stdio.h>
#include<stddef.h>
struct S3
{double d;char c;int i;
};
struct S4
{char c1;struct S3 s3;double d;
};
int main()
{printf("%d\n", sizeof(struct S4));return 0;
}

 

为什么要对齐

参考大部分资料:

  • 平台原因(移植原因): 不是所有的硬件平台都能访问任意地址上的任意数据的;某些硬件平台只能在某些地址处取某些特定类型的数据,否则抛出硬件异常。
  • 性能原因: 数据结构(尤其是栈)应该尽可能地在自然边界上对齐。 原因在于,为了访问未对齐的内存,处理器需要作两次内存访问;而对齐的内存访问仅需要一次访问。
  • 总体来说:结构体的内存对齐是拿空间来换取时间的做法

优化结构体成员顺序 

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

//例如:
struct S1
{char c1;//1int i;//1char c2;//4
};//8
struct S2
{char c1;//1char c2;//4int i;//1
};//12

 S1和S2类型的成员一模一样,但是S1和S2所占空间的大小有了一些区别。

修改默认对齐数

结构在对齐方式不合适的时候,我们可以自己更改默认对齐数。一般设置默认对齐数:2的次方

#include <stdio.h>
#pragma pack(8)//设置默认对齐数为8
struct S1
{char c1;int i;char c2;
};
#pragma pack()//取消设置的默认对齐数,还原为默认
#pragma pack(1)//设置默认对齐数为1
struct S2
{char c1;int i;char c2;
};
#pragma pack()//取消设置的默认对齐数,还原为默认
int main()
{//输出的结果是什么?printf("%d\n", sizeof(struct S1));printf("%d\n", sizeof(struct S2));return 0;
}

 大家自己动一动手画一画图,思考答案!

百度笔试题

写一个宏,计算结构体中某变量相对于首地址的偏移,并给出说明
考察: offsetof 宏的实现
注:这里还没学习宏,后面博文讲解。

✔✔✔✔✔最后,感谢大家的阅读,若有错误和不足,欢迎指正!

下篇博文我们继续自定义类型。

代码------→【gitee:唐棣棣 (TSQXG) - Gitee.com】

联系------→【邮箱:2784139418@qq.com】

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

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

相关文章

二十九、高级IO与多路转接之epollreactor(收官!)

文章目录 一、Poll&#xff08;一&#xff09;定义&#xff08;二&#xff09;实现原理&#xff08;三&#xff09;优点&#xff08;四&#xff09;缺点 二、I/O多路转接之epoll&#xff08;一&#xff09;从网卡接收数据说起&#xff08;二&#xff09;如何知道接收了数据&…

蓝桥杯每日一题2023.9.28

AcWing 4409. 砍竹子 - AcWing 题目描述 题目分析 注&#xff1a;sqrtl的范围为long double&#xff0c;比sqrt更加精确 使用优先队列维护一段区间&#xff0c;如果连续一段相同就合并为一个区间&#xff0c;从大到小去枚举&#xff0c;每次先取出最大的一段&#xff0c;双…

专业综合课程设计 - 优阅书城项目(第一版)

此项目是《专业综合课程设计》带练项目 实现的功能有&#xff1a; 登录、注销、添加图书、删除图书、编辑图书 包含资源&#xff1a; 优阅书城&#xff08;bookstore&#xff09;源码 数据库数据 课程笔记 下载链接&#xff1a;https://wwpv.lanzoue.com/i79nx1av4doj 登录功…

windows系统服务管理命令sc

sc可以用于管理系统服务、计划任务、系统日志等方面&#xff0c;是不可或缺的神器。 基本用法 在命令提示符下输入sc命令&#xff0c;然后按回车键。 上图展示的是sc命令的使用方法&#xff0c;支持哪些参数实现哪些功能 要查看系统所有服务列表&#xff0c;包括它们是否正在…

逻辑回归评分卡

文章目录 一、基础知识点(1)逻辑回归表达式(2)sigmoid函数的导数损失函数(Cross-entropy, 交叉熵损失函数)交叉熵求导准确率计算评估指标 二、导入库和数据集导入库读取数据 三、分析与训练四、模型评价ROC曲线KS值再做特征筛选生成报告 五、行为评分卡模型表现总结 一、基础知…

Go-Python-Java-C-LeetCode高分解法-第八周合集

前言 本题解Go语言部分基于 LeetCode-Go 其他部分基于本人实践学习 个人题解GitHub连接&#xff1a;LeetCode-Go-Python-Java-C 欢迎订阅CSDN专栏&#xff0c;每日一题&#xff0c;和博主一起进步 LeetCode专栏 我搜集到了50道精选题&#xff0c;适合速成概览大部分常用算法 突…

2023年-华为机试题库B卷(Python)【满分】

1️⃣ 2️⃣ 3️⃣ 4️⃣ 5️⃣ 6️⃣ 7️⃣ 8️⃣ 9️⃣ &#x1f51f; 华为机试题库B卷 已于5月10号 更新为2023 B卷 &#xff08;2023-10-04 更新本文&#xff09; 华为机试有三道题目&#xff0c;前两道属于简单或中等题&#xff0c;分值为100分&#xff0c;第三道为中等或…

互动营销类游戏开发小程序游戏开发

互动营销类游戏开发是一种创意和有趣的方式&#xff0c;可以用于推广产品、品牌或服务。这些游戏旨在吸引玩家并与他们互动&#xff0c;以促使他们与您的品牌建立更紧密的联系。 当今数字时代的市场竞争激烈&#xff0c;品牌和企业需要采用创新的方法来吸引并保持用户的兴趣。…

MySQL基础-事务

目录 1.事务简介 2.事务的操作 2.1 实验需要用到的数据 2.2 完成转账操作 修改事务执行方式 手动开启事务的方式 3.事务的四大特性 4.并发事务问题 5.事务隔离级别 5.1 事务隔离级别分类 5.2 查看事务隔离级别 5.3 设置事务隔离级别 1.事务简介 事务是一组操作的集合…

实时人脸五观检测:基于libfacedetection(CNN模型)

一、前言 随着人工智能技术的不断发展,人脸检测已成为计算机视觉领域的重要应用之一。人脸检测是一种将输入图像中的人脸位置和轮廓提取出来的技术,广泛应用于人脸识别、智能监控、人机交互等领域。利用libfacedetection开源的人脸检测库,实现人脸检测。 libfacedetection…

第一课数组、链表、栈、队列

第一课数组、链表、栈、队列 acwing136 邻值查找---中等题目描述代码展示 lc20.有效的括号--简单题目描述代码展示 lc25.K 个一组翻转链表--困难题目描述代码展示 lc26.删除有序数组中的重复项--简单题目描述代码展示 lc88.合并两个有序数组--简单题目描述代码展示 lc141.环形链…

mysql-binlog

1. 常用的binlog日志操作命令 1. 查看bin-log是否开启 show variables like log_%;2. 查看所有binlog日志列表 show master logs;3.查看master状态 show master status;4. 重置&#xff08;清空&#xff09;所有binlog日志 reset master;2. 查看binlog日志内容 1、使用mysqlb…

目前制造企业生产计划现状是什么?有没有自动化排产系统?

大家都知道&#xff0c;人的指挥中心是大脑&#xff0c;大脑对我们的发出各种各样的指令&#xff0c;告诉我们&#xff1a;“手”做什么事情&#xff0c;“眼睛”看什么地方&#xff0c;“耳朵”听什么声音&#xff0c;然后再将摸到的、看到的、听到的信息传递给大脑&#xff0…

《视觉 SLAM 十四讲》V2 第 4 讲 李群与李代数 【什么样的相机位姿 最符合 当前观测数据】

P71 文章目录 4.1 李群与李代数基础4.1.3 李代数的定义4.1.4 李代数 so(3)4.1.5 李代数 se(3) 4.2 指数与对数映射4.2.1 SO(3)上的指数映射罗德里格斯公式推导 4.2.2 SE(3) 上的指数映射SO(3),SE(3),so(3),se(3)的对应关系 4.3 李代数求导与扰动模型4.3.2 SO(3)上的李代数求导…

Vue中如何进行图像处理与图像滤镜

在Vue中进行图像处理与图像滤镜 图像处理和滤镜效果是现代Web应用程序中常见的功能之一。Vue.js作为一个流行的JavaScript框架&#xff0c;为实现这些功能提供了许多工具和库。本文将介绍如何使用Vue来进行图像处理与图像滤镜&#xff0c;包括使用HTML5 Canvas和CSS滤镜。 准备…

Nacos与Eureka的区别

大家好我是苏麟今天说一说Nacos与Eureka的区别. Nacos Nacos的服务实例分为两种l类型&#xff1a; 临时实例&#xff1a;如果实例宕机超过一定时间&#xff0c;会从服务列表剔除&#xff0c;默认的类型。非临时实例&#xff1a;如果实例宕机&#xff0c;不会从服务列表剔除&…

网络层·IP协议

承接前文TCP协议-CSDN博客 简介 协议头格式 网段划分(重要) 划分方法 IP地址的数量限制(背景介绍) 私有IP地址和公网IP地址(提出解决思路) NAT技术(解决方法) 路由 网络层 在复杂的网络环境中确定一个合适的路径 IP协议 主机: 配有IP地址, 可以认为就是你的电脑; 路由器:…

分享几个优秀开源免费管理后台模版,建议收藏!

大家好&#xff0c;我是 jonssonyan 今天和大家分享一些免费开源的后台管理页面&#xff0c;帮助大家快速搭建前端页面。为什么要用模板&#xff1f;道理很简单&#xff0c;原因是方便我们快速开发。我们不应该花太多的时间在页面调整上&#xff0c;而应该把精力放在核心逻辑和…

【有限域除法】二元多项式除法电路原理及C语言实现

二元多项式除法电路原理 例: g ( x ) = x 4 + x 2 + x + 1 g(x)=x^4 + x^2+x+1

Linux系统编程系列之进程间通信-IPC对象

Linux系统编程系列&#xff08;16篇管饱&#xff0c;吃货都投降了&#xff01;&#xff09; 1、Linux系统编程系列之进程基础 2、Linux系统编程系列之进程间通信(IPC)-信号 3、Linux系统编程系列之进程间通信(IPC)-管道 4、Linux系统编程系列之进程间通信-IPC对象 5、Linux系统…