初步认识结构体

news2024/9/29 23:33:26

hello,hello,各位小伙伴,本篇文章跟大家一起学习结构体,并跟大家一边做题一边进行学习和理解。感谢大家对我上一篇的支持,如有什么问题,还请多多指教!

如果本篇文章对你有帮助,还请各位点点赞!!! 

2ba8a98057f84c00a921517ba5e63a41.jpeg

话不多说,正题开始

关于结构体的地定义和初始化在操作符详解有详细讲解

 所以本文章将会从继后面继续深入讲解,如果小伙伴忘记了可以先去回顾一下

1.匿名结构体

所谓匿名就是不显示名字,但这里的匿名有点不一样,是完全没有名字,看个例子:

struct
{
 int a;
 char b;
 float c;
}x;

struct
{
 int a;
 char b;
 float c;
}a[20], *p;

但是匿名结构体只能使用一次,并且并且在创建变量时只能够像上述例子一样创建(毕竟没有名字):

匿名的结构体类型,如果没有对结构体类型重命名的话,基本上只能使用一次。

匿名结构体都没有名字(类型),那是不是所有匿名结构体的类型都一样啊?

按照上述例子,请看如下代码:

p = &x;

这样写合法吗?答案是:不!

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

2.结构体自引用

相比在学习结构体时会有小伙伴想,如果在结构体成员列表里引用结构体本身会怎样?

这就是结构体自引用,如:

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

但是再仔细想一下,这样写真的可以吗?这样写似乎在不断的创建结构体,永无止境了

没错,这样写是不行的,如果想引用结构体本身,并不一定要在结构体里创建结构体,我们学过的指针可以帮助我们访问结构体自身,正确写法应该是这样:

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

再来看看如下代码:

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

在结构体自引用时对匿名结构体重命名,这样写可行吗?

答案是否定的,正如上述例子,Node是匿名结构体重命名后的名字,但是上述代码在重命名之前就已经使用了Node,那么编译器就不认识这个Node,就会报错,解决这个问题的方案就是:定义结构体不要使用匿名结构体了

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

关于结构体的基本使用就结束了,接下来就是如何计算结构体的大小了

3. 结构体的大小

int的大小为4个字节,short为2个字节....那么结构体呢?这就有点复杂了,我们一步一步来

3.1 对齐规则

首先我们要明白对齐规则:

1. 结构体的第⼀个成员对齐到和结构体变量起始位置偏移量为0的地址处

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

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

- VS 中默认的值为 8

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

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

4. 如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最⼤对⻬数的整数倍处,结构 体的整体⼤大小就是所有最大对齐数(含嵌套结构体中成员的对⻬数)的整数倍。

干说难懂,上代码:

注意:这了所用的编译器为VS2022

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

 该结构体大小为多少呢?看图解:

好,我们再来看如下练习:

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

 该结构体大小为多少呢?看图解:

struct S3
{
 double d;
 char c;
 int i;
};

该结构体大小为多少呢?看图解:

再来看如下练习:打印结果是什么呢?

struct S4
{
 char c1;
 struct S3 s3;
 double d;
};
printf("%d\n", sizeof(struct S4));

答案是:32

3.2 为什么存在内存对齐?

⼤部分的参考资料都是这样说的:

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

2. 性能原因: 数据结构(尤其是栈)应该尽可能地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要 作两次内存访问;而对齐的内存访问仅需要⼀次访问。假设⼀个处理器总是从内存中取8个字节,则地 址必须是8的倍数。如果我们能保证将所有的double类型的数据的地址都对齐成8的倍数,那么就可以 用⼀个内存操作来读或者写值了。否则,我们可能需要执行两次内存访问,因为对象可能被分放在两 个8字节内存块中。

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

3.2.1 如何节省空间? 

我们来看看如下代码:

struct S1
{
 char c1;
 int i;
 char c2;
};
struct S2
{
 char c1;
 char c2;
 int i;
};

通过计算,这两个结构体虽然成员列表的内容相同,但是二者的大小并不一样,struct S2 比  struct S1更小,所以节省空间方法:

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

 4. 修改默认对齐数

当结构体在对齐方式不合理时,我们可以自己更改默认对齐数

#pragma 这个预处理指令,可以改变编译器的默认对齐数

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

输出结果是:6

根据选取对齐数规则:对齐数 = 编译器默认的⼀个对齐数与该成员变量大小的较小值

因为我们默认对齐数被修改成1,而1是最小的正整数,所以我们会一直以1作为对齐数 

5. 结构体传参 

 传参我们有学过传值传参和传址传参两种方式,同样,结构体传参也有这两种方式,那么这两种有什么区别呢?哪种有会更好呢?看如下代码:

struct S
{
 int data[1000];
 int num;
};
struct S s = {{1,2,3,4}, 1000};
//结构体传参
void print1(struct S s)
{
 printf("%d\n", s.num);
}
//结构体地址传参
void print2(struct S* ps)
{
 printf("%d\n", ps->num);
}
int main()
{
 print1(s); //传结构体
 print2(&s); //传地址
 return 0;
}

打印结果都一样,但是是否二者都一样呢?答案是否定的

上面的 print1 和 print2 函数哪个好些? 答案是:首选print2函数。

我们知道,传值传参中的形参,是实参的一份临时拷贝,而传址传参是直接访问该结构体地址,不需要再创建临时变量,所以速度上会比传值传参更快

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

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

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

6. 结构体实现位段 

注意:以下测试全部在VS2022环境下实现

6.1 什么是位段 

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

1. 位段的成员必须是 int、unsigned int 或signed int ,在C99中位段成员的类型也可以 选择其他类型。

2. 位段的成员名后边有⼀个冒号和⼀个数字。

struct A
{
 int a:2;
 int b:5;
 int c:10;
 int d:30;
};

 A就是⼀个位段类型。 那位段A所占内存的⼤⼩是多少?答案是:8字节

其实位段就是为了节省空间的,冒号后面的数字就是对应变量占多少个比特位

2+5+10+30 = 47,这么看来是不是非常节省空间,那是否就可以一无止境的节省?

那肯定不是,从答案是8字节就可以看出,并非是无止境的节省空间,47个比特位用6个字节即可,答案却是8字节,这和位段的内存分配有关

6.2 位段的内存分配 

1. 位段的成员可以是 int unsigned int signed int 或者是 char 等类型

2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的⽅式来开辟的。

3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使⽤位段。 

第2点的意思就是:比如int 类型不够存储时,内存会以4个字节(int 类型占4个字节)来开辟新内存,当char 类型不够存储时,内存会以1个字节(char 类型占1个字节)开辟新空间,干说难懂,上代码:

就如上一题

struct A
{
 int a:2;
 int b:5;
 int c:10;
 int d:30;
};

 好,我们来看一下练习:

struct A
{
 char a:2;
 char b:5;
 char c:7;
 char d:4;
};

那我们加点料:

struct S
{
 char a:2;
 char b:5;
 char c:7;
 char d:4;
};
struct S s = {0};
s.a = 10;
s.b = 12;
s.c = 3;
s.d = 4;

 空间是如何开辟的?我们先分析一下:(VS2022环境下测试)

答案正确 

6.3 位段的跨平台问题 

1. int 位段被当成有符号数还是⽆符号数是不确定的。

2. 位段中最⼤位的数⽬不能确定。(16位机器最⼤16,32位机器最⼤32,写成27,在16位机器出问题。

3. 位段中的成员在内存中从左向右分配,还是从右向左分配标准尚未定义。(取决于编译器)

4. 当⼀个结构包含两个位段,第⼆个位段成员⽐较⼤,⽆法容纳于第⼀个位段剩余的位时,是舍弃剩余的位还是利⽤,这是不确定的。(取决于编译器)

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

6.4 位段的应用 

下图是⽹络协议中,IP数据报的格式,我们可以看到其中很多的属性只需要⼏个bit位就能描述,这⾥ 使⽤位段,能够实现想要的效果,也节省了空间,这样⽹络传输的数据报⼤⼩也会较⼩⼀些,对⽹络 的畅通是有帮助的。

 6.5 位段使用的注意事项

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

举个例子:

#include<stdio.h>
struct S
{
 char a:2;
 char b:5;
 char c:7;
 char d:4;
}s;
int main()
{
    scanf("%d",&s.c);//是错误的
    //这才是正确写法    
    s.c = 10;
    return 0;
}

好啦,本章对于结构体的学习就先到这里,如果有什么问题,还请指教指教,希望本篇文章能够对你有所帮助,我们下一篇见!!! 

如你喜欢,点点赞就是对我的支持,感谢感谢!!!

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

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

相关文章

H5流媒体播放器EasyPlayer播放H.265新增倍速播放功能,具体如何实现?

目前我们TSINGSEE青犀视频所有的视频监控平台&#xff0c;集成的都是EasyPlayer.js版播放器&#xff0c;它属于一款高效、精炼、稳定且免费的流媒体播放器&#xff0c;可支持多种流媒体协议播放&#xff0c;包括WebSocket-FLV、HTTP-FLV&#xff0c;HLS&#xff08;m3u8&#x…

智能优化算法应用:基于蛾群算法无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于蛾群算法无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于蛾群算法无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.蛾群算法4.实验参数设定5.算法结果6.参考文献7.MATLAB…

大模型技术的发展与实践

一、大模型的概念 大型语言模型&#xff0c;也称大语言模型、大模型&#xff08;Large Language Model&#xff0c;LLM&#xff1b;Large Language Models&#xff0c;LLMs) 。 大语言模型是一种深度学习模型&#xff0c;特别是属于自然语言处理&#xff08;NLP&#xff09;的…

二叉树(判断是否为平衡二叉树)

题目&#xff08;力扣&#xff09;&#xff1a; 观察题目&#xff0c;发现最重要的条件就是&#xff0c;两颗子树的高度差的绝对值不超过1&#xff0c;我们就可以用递归将所有左子树和右子树都遍历一个&#xff0c;求出他们的高度差&#xff0c;若差值 > 1&#xff0c;则返回…

C语言数组(上)

# 数组的概念 数组是一组相同类型元素的集合。数组中存放的是一个或多个数据&#xff0c;但是数组中的元素个数不能为零&#xff1b;数组中存放的所有元素&#xff08;数据&#xff09;的类型必须是相同的。 数组分为一维数组和多维数组&#xff0c;多维数组一般比较多见的是二…

案例研究|作为一家BI厂商,飞致云是如何人人使用DataEase的?

杭州飞致云信息科技有限公司&#xff08;以下简称为飞致云&#xff09;长期秉持“软件用起来才有价值&#xff0c;才有改进的机会”的核心价值观&#xff0c;以“为数字经济时代创造好软件”为使命&#xff0c;致力于成为中国数字化团队首选的通用工具软件提供商。在软件产品不…

侯捷C++八部曲(一,面向对象)

头文件和类的声明 inline inline修饰函数&#xff0c;是给编译器的一个建议&#xff0c;到底是否为inline由编译器来决定&#xff0c;inline修饰的函数在使用时是做简单的替换&#xff0c;这样就避免了一些函数栈空间的使用&#xff0c;从能提升效率。从另一种角度看&#xff…

Linux lshw命令(lshw指令)(List Hardware,获取底层硬件信息)(查询硬件信息)

文章目录 Linux lshw命令&#xff1a;一个全面的硬件信息查询工具介绍安装lshw使用lshwlshw的选项和参数lshw文档英文文档中文文档 命令示例lshw -c network -sanitize查看系统网络硬件信息&#xff0c;并移除敏感项&#xff08;显示为REMOVED&#xff09; lshw与其他命令的对比…

个人Windows电脑通过Cloudreve+Cpolar搭建PHP云盘系统公网可访问

文章目录 1、前言2、本地网站搭建2.1 环境使用2.2 支持组件选择2.3 网页安装2.4 测试和使用2.5 问题解决 3、本地网页发布3.1 cpolar云端设置3.2 cpolar本地设置 4、公网访问测试5、结语 1、前言 自云存储概念兴起已经有段时间了&#xff0c;各互联网大厂也纷纷加入战局&#…

10个顶级Linux开源反向代理服务器 - 解析与导航

反向代理服务器是一种部署在客户端和后端/源服务器之间的代理服务器&#xff0c;例如 NGINX、Apache 等 HTTP 服务器或用 Nodejs、Python、Java、Ruby 编写的应用程序服务器、PHP 和许多其他编程语言。 它是一个网关或中间服务器&#xff0c;它接受客户端请求&#xff0c;将其传…

gitlab-jenkins-shell-helm-chart-k8s自动化部署微服务

1.准备好编译环境的容器&#xff0c;所有容器的镜像制作在gemdale-dockerfile这个代码库里面&#xff0c;也可以直接拉取官方镜像部署 docker run --name node1420-patternx -v /data/var/www/:/data/var/www/ -v /var/jenkins_home/:/var/jenkins_home/ -v /mnt/hgfs/:/mnt/h…

SpringBoot自定义异常处理机制

说明&#xff1a;在完整的项目结构中&#xff0c;我们通常会创建一个自定义的异常处理机制&#xff0c;在系统可能出现异常的地方手动抛出这些异常&#xff0c;可以快速定位到异常代码片段&#xff0c;提供项目的可维护性。 本文介绍在SpringBoot项目中&#xff0c;搭建一套自…

万字长文带你搞定MMUTLBTWU

最近一直在学习内存管理&#xff0c;也知道MMU是管理内存的映射的逻辑IP&#xff0c;还知道里面有个TLB。 今天刚刚好看到了几篇前辈的文章&#xff0c;很是不错&#xff0c;于是这里来一起学习一下吧。 PART 一&#xff1a;MMU 架构篇 MMU&#xff08;Memory Management Uni…

解决在Linux中进行redis的主从复制时出现的从机可以获取到主机的信息,主机获取不到从机的信息~

主机&#xff1a; 从机1&#xff1a; 从机2&#xff1a; 出现上述的原因是我在redis.conf中设置了密码&#xff0c;那么就导致了我在进行主从复制时&#xff0c;需要进行密码验证&#xff0c;然后我在网上查阅了很多资料&#xff0c;有的说让在从机中指定密码&#xff0c;有的说…

12.04 二叉树中等题

513. 找树左下角的值 给定一个二叉树的 根节点 root&#xff0c;请找出该二叉树的 最底层 最左边 节点的值。 假设二叉树中至少有一个节点。 示例 1: 输入: root [2,1,3] 输出: 1 思路&#xff1a;找到最低层中最左侧的节点值&#xff0c;比较适合层序遍历&#xff0c;返回最…

按升序输出各个字符串。

编写程序&#xff0c;设计并实现如下功能&#xff1a;从键盘输入多个字符串&#xff0c;按升序输出各个字符串。

笔记66:自注意力和位置编码

本地笔记地址&#xff1a;D:\work_file\&#xff08;4&#xff09;DeepLearning_Learning\03_个人笔记\3.循环神经网络\第10章&#xff1a;动手学深度学习~注意力机制 a a a a a a a a a a a a a a a a a a a

YOLOv8-Seg改进:简单高效的模块-现代反向残差移动模块 (iRMB) | | ICCV2023 EMO

🚀🚀🚀本文改进:设计了一种面向移动端应用的简单而高效的现代反向残差移动模块 (Inverted Residual Mobile Block, iRMB),它吸收了类似 CNN 的效率来模拟短距离依赖和类似 Transformer 的动态建模能力来学习长距离交互,引入YOLOV8 🚀🚀🚀YOLOv8-seg创新专栏:h…

艾瑞:央国企数字化升级,低代码首选得帆云!

中国权威咨询机构艾瑞咨询最新发布了《2023年央国企数字化升级研究报告》。 THE NEW RESEARCH 报告认为 央国企作为中国特色社会主义的重要物质基础和政治基础&#xff0c;肩负着推动经济发展和增强社会价值的重要责任&#xff0c;必须在数字化升级中发挥引领作用。当前&#…

5.【自动驾驶与机器人中的SLAM技术】2D点云的scan matching算法 和 检测退化场景的思路

目录 1. 基于优化的点到点/线的配准2. 对似然场图像进行插值&#xff0c;提高匹配精度3. 对二维激光点云中会对SLAM功能产生退化场景的检测4. 在诸如扫地机器人等这样基于2D激光雷达导航的机器人&#xff0c;如何处理悬空/低矮物体5. 也欢迎大家来我的读书号--过千帆&#xff0…