结构体+结构体内存对齐+结构体实现位段

news2024/10/7 4:31:45

结构体+内存对齐+实现位段

  • 一.结构体
    • 1.结构体的声明
    • 2.结构体变量成员访问操作符
    • 3.结构体传参
    • 4.匿名结构体
    • 5.结构的自引用
  • 二.结构体内存对齐
    • 1.对齐规则
    • 2.为什么存在内存对齐?
    • 3.修改默认对齐数
  • 三.结构体实现位段
    • 1.什么是位段
    • 2.位段的内存分配
    • 3.位段的跨平台问题
    • 4.位段的应用
    • 5.位段使用的注意事项

前言:

  1. 学习了数组后发现数组中的元素只能是相同类型的变量,那么有没有可以存放不同类型的变量呢?
  2. 结构体:一些值的集合,这些值称为成员变量,结构体的每个成员可以是不同类型的变量

一.结构体

1.结构体的声明

struct tag
{
	member-list;//结构体成员列表
}variable-list;//结构体变量列表

例如:描述一个人

struct Person {
    int age;//年龄
    char name[50];//姓名
    float height;//身高
};//封号不能丢

2.结构体变量成员访问操作符

  1. 结构体变量.结构体成员名。
  2. 结构体指针变量->结构体成员名。
#include <stdio.h>
struct Person
{
    int age;
    char name[50];
    float height;
}p1 = { 20,"zhangsan",185.5 }, * ps;//全局变量(*ps:结构体指针ps)

int main()
{
    struct Person p2 = { 18,"lisi",173.2 };//局部变量
    struct Person p3 = { 19,"wangwu",180.8 };//局部变量
    ps = &p3;
    printf("%d %s %.1f\n", p1.age, p1.name, p1.height);//结构体成员访问操作符:.
    printf("%d %s %.1f\n", p2.age, p2.name, p2.height);

    printf("%d %s %.1f\n", (*ps).age, (*ps).name, (*ps).height);
    printf("%d %s %.1f\n", ps->age, ps->name, ps->height);//结构体成员访问操作符:->等价于先*再.
    return 0;
}

在这里插入图片描述

3.结构体传参

  1. 传结构体。
  2. 传结构体的地址。
#include <stdio.h>
struct Person
{
    int age;
    char name[50];
    float height;
};
void test1(struct Person p)//用结构体接收
{
    printf("%d %s %.1f\n", p.age, p.name, p.height);
}
void test2(struct Person* p)//用结构体指针接收
{
    printf("%d %s %.1f\n", p->age, p->name, p->height);
}
int main()
{
    struct Person p1 = { 20,"zhangsan",185.5 };
    test1(p1);//传结构体
    test2(&p1);//传结构体的地址
    return 0;
}

在这里插入图片描述

思考:我们发现二者都可以成功访问结构体成员,那二者有什么区别呢?

  1. 传递结构体时:其实函数内部创建了一个临时结构体变量存放传入的结构体,当结构体很大时会额外占用空间不划算。(本质上是值传递)。
  2. 传递结构体地址时:只需创建4个字节结构体指针变量,通过其来访问结构体成员,可以大大节省空间。(本质上是地址/指针传递)。
  3. 推荐传递结构体地址

在这里插入图片描述

4.匿名结构体

//匿名结构体类型 
struct//不完全声明,由于没有名字,无法在其之后创建变量
{
    int age;
    char name[50];
    float height;
}s1, s2;//在结构体声明的时候直接创建变量,不能在其之后创建变量了,只能使用一次
int main()
{
	struct s3;//error
}
  • 当只需使用一次可以使用(在声明结构体时,直接创建变量,不能在其之后创建变量了)。

思考:以下代码行不行

struct
{
    int age;
    char name[50];
    float height;
}s1;
struct
{
    int age;
    char name[50];
    float height;
}*ps;

int main()
{	
    ps = &s1;//?
	return 0;
}
  • 答案:不行,看似一样,其实这两个结构体是不同类型的,只是成员变量相同的不同结构体类型,二者不兼容。(没有名字导致的问题)。

5.结构的自引用

比如:定义一个链表的节点

struct Node
{
 	int data;//存放数据
 	struct Node* next;//存放指针
};

二.结构体内存对齐

注意:面试时计算结构体的大小是一个热门的考点,一定要学会。

1.对齐规则

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

offsetof宏:计算结构体成员相较于结构体变量起始位置的偏移量,头文件stddef.h
在这里插入图片描述

例如:计算结构体大小的代码。

#include<stdio.h>
#include<stddef.h>
struct S1
{
    char c1;//自身大小1,默认对齐数8,对齐数1
    char c2;//自身大小1,默认对齐数8,对齐数1
    int n;//自身大小4,默认对齐数8,对齐数4
};
struct S2
{
    char c1;//自身大小1,默认对齐数8,对齐数1
    int n;//自身大小4,默认对齐数8,对齐数4
    char c2;//自身大小1,默认对齐数8,对齐数1
};
int main()
{
    printf("%zd\n", offsetof(struct S1, c1));//0
    printf("%zd\n", offsetof(struct S1, c2));//1
    printf("%zd\n", offsetof(struct S1, n));//4
    printf("%zd\n", sizeof(struct S1));//8

    printf("%zd\n", offsetof(struct S2, c1));//0
    printf("%zd\n", offsetof(struct S2, n));//4
    printf("%zd\n", offsetof(struct S2, c2));//8
    printf("%zd\n", sizeof(struct S2));//12
    return 0;
}

在这里插入图片描述

练习:

#include<stdio.h>
struct S1
{
    double d;//自身大小8,默认对齐数8,对齐数8
    char c;//自身大小1,默认对齐数8,对齐数1
    int i;//自身大小4,默认对齐数8,对齐数4
};
struct S2
{
    char c1;//自身大小1,默认对齐数8,对齐数1
    struct S1 s1;//自身大小16,默认对齐数8,对齐数8
    //如果嵌套了结构体的情况,嵌套的结构体成员对齐到《自己的成员中最大对齐数的整数倍处(d的对齐数的整数倍处)》,
    //结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。
    double d;//自身大小8,默认对齐数8,对齐数8
};
int main()
{
    printf("%zd\n", sizeof(struct S1));//16
    printf("%zd\n", sizeof(struct S2));//32
    return 0;
}

在这里插入图片描述

2.为什么存在内存对齐?

在这里插入图片描述

在这里插入图片描述

  • 那在设计结构体的时候,我们既要满足对齐,又要节省空间,如何做到:
//例如: 
#include<stdio.h>
struct S1
{
	char c1;
	int i;
	char c2;
};
struct S2
{
	char c1;
	char c2;
	int i;
};
int main()
{
	printf("%zd\n", sizeof(struct S1));//12
	printf("%zd\n", sizeof(struct S2));//8
	//S1 和 S2 类型的成员⼀模⼀样,但是 S1 和 S2 所占空间的大小有了⼀些区别。
	return 0;
}

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

3.修改默认对齐数

  1. VS上默认对齐数为8。
  2. #pragma pack(一般为2^n) 这个预处理指令,可以改变编译器的默认对齐数。
  3. 例如#pragma pack(1),#pragma pack(2),#pragma pack(4)。
  4. #pragma pack() == #pragma pack(8)。
#include<stdio.h>
#pragma pack(1)//修改默认对齐数变成1
struct S
{
	char c1;//自身大小1,默认对齐数1,对齐数1
	int i;//自身大小4,默认对齐数1,对齐数1
	char c2;//自身大小1,默认对齐数1,对齐数1
};
#pragma pack()//将默认对齐数修改为8
int main()
{
	printf("%zd\n", sizeof(struct S));//6
	return 0;
}

三.结构体实现位段

  • 结构体有实现位段的功能。

1.什么是位段

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

  1. 位段的成员必须是 int、unsigned int 或 signed int ,在C99中位段成员的类型也可以选择其他类型。
  2. 位段的成员名后边有一个冒号和一个数字。
  3. 位段中的位:二进制的位。

位段与结构体语法上的区别,代码如下:

#include<stdio.h>
struct A//结构体
{
	int a;
	int b;
	int c;
	int d;
};
struct B//结构体实现位段
{
	int a : 2;//只给了两个比特位,意味着只能存放0,1,2,3,不能存放大于它们的值
	int b : 5;//同理
	int c : 10;
	int d : 30;
};
int main()
{
	printf("%zd\n", sizeof(struct A));//16个字节
	printf("%zd\n", sizeof(struct B));//8个字节
	//发现位段较于结构体节省了空间
	return 0;
}
  • 总结:位段相较于结构体节省了空间。

2.位段的内存分配

  1. 位段的成员可以是 int unsigned int signed int 或者是 char 等类型。
  2. 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。
  3. 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。
#include<stdio.h>
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("%zd\n", sizeof(struct S));//3个字节
	return 0;
}

1.给定了空间后,在空间的内部是从左向右使用,还是从右向左使用,这个是不确定的。
 假设:从右向左使用。
2.当剩下的空间不足以存放下一个成员的时候,空间是浪费还是使用,这个是不确定的。
 假设:浪费。

在这里插入图片描述

在这里插入图片描述

3.位段的跨平台问题

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

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

4.位段的应用

  • 下图是网络协议中,IP数据报的格式,我们可以看到其中很多的属性只需要几个bit位就能描述,这里使用位段,能够实现想要的效果,也节省了空间,这样网络传输的数据报大小也会较小⼀些,对网络的畅通是有帮助的。
    在这里插入图片描述
  1. 在网络中发送数据的时候,需要进行数据的封装,例如:加上源地址与目的地址。(计算机网络中的网络层协议——> IP协议)。
  2. 为了避免网络拥堵,相办法节省空间,使用的就是位段。

5.位段使用的注意事项

  1. 位段的几个成员共有同⼀个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置处是没有地址的。内存中每个字节分配一个地址,一个字节内部的bit位是没有地址的。
  2. 所以不能对位段的成员使用&操作符,这样就不能使用 scanf 直接给位段的成员输⼊值,只能是先输⼊放在一个变量中,然后赋值给位段的成员。
#include<stdio.h>
struct A
{
	int _a : 2;
	int _b : 5;
	int _c : 10;
	int _d : 30;
};
int main()
{
	//这是错误的
	struct A sa = { 0 };
	scanf("%d", &sa._b); 

	//正确的示范
	int b = 0;
	scanf("%d", &b);
	sa._b = b;
	return 0;
}

创作不易,如果能帮到你的话能赏个三连吗?感谢啦!!!

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

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

相关文章

C++标准模板(STL)- 迭代器库-迭代器适配器 - 逆序遍历的迭代器适配器

迭代器库-迭代器适配器 迭代器库提供了五种迭代器的定义&#xff0c;同时还提供了迭代器特征、适配器及相关的工具函数。 迭代器分类 迭代器共有五 (C17 前)六 (C17 起)种&#xff1a;遗留输入迭代器 (LegacyInputIterator) 、遗留输出迭代器 (LegacyOutputIterator) 、遗留向…

2024年城市建设与环境管理国际会议(ICUCEM 2024)

2024 International Conference on Urban Construction and Environmental Management 【1】大会信息 大会地点&#xff1a;中国成都 投稿邮箱&#xff1a;icucemsub-paper.com 【2】会议简介 2024年城市建设与环境管理国际会议是一个专注于探讨城市建设与环境管理前沿议题…

#02 安装指南:如何配置Stable Diffusion环境

文章目录 前言前置条件第1步&#xff1a;安装Python和PIP第2步&#xff1a;创建虚拟环境第3步&#xff1a;安装PyTorch和CUDA第4步&#xff1a;安装Stable Diffusion相关库第5步&#xff1a;测试环境结论 前言 在之前的文章中&#xff0c;我们介绍了Stable Diffusion基础入门和…

【ArcGIS微课1000例】0114:基于DEM地形数据整体抬升或下降高程

相关阅读:【GlobalMapper精品教程】083:基于DEM整体抬升或下降地形高程的两种方式 文章目录 一、任务分析二、栅格计算器简介三、地形整体修改四、注意事项一、任务分析 打开软件,加载配套实验数据中的0112.rar中的dem数据,如下所示,dem的高程范围为256.75~342.37米,现在…

QT之动态加载树节点(QTreeWidget)

之前写过一篇动态加载ComboBox&#xff0c;可参见下面这篇文章 QT之动态加载下拉框&#xff08;QComboBox&#xff09; 同理QTreeWidget也可以实现动态加载&#xff0c;在一些异步加载数据&#xff0c;并且数据加载比较耗时&#xff0c;非常实用。 效果 原理分析 要实现此类效…

618精选网络安全书单:打造数字世界的钢铁长城!

文章目录 《内网渗透实战攻略》《Kali Linux高级渗透测试&#xff08;原书第4版&#xff09;》《CTF那些事儿》《权限提升技术&#xff1a;攻防实战与技巧》《数字政府网络安全合规性建设指南&#xff1a;密码应用与数据安全》《红蓝攻防&#xff1a;构建实战化网络安全防御体系…

ECharts 图形化看板 模板(简单实用)

目录 一、官网 二、模板 ①定义请求​编辑 ② 将请求统一管理&#xff0c;别的页面引用多个请求时更便于导入。​编辑 ③最终模板 三、执行效果 四、后端代码 4.1 controller 4.2 xml 4.3 测试接口 一、官网 获取 ECharts - 入门篇 - 使用手册 - Apache ECharts 二、…

医疗器械网络安全风险管理的基本步骤

医疗器械网络安全风险管理是一个复杂的过程&#xff0c;涉及到多个环节和步骤。以下是一些基本的步骤和关键点&#xff1a; 风险识别&#xff1a;首先需要对医疗器械的软件、网络连接和通信协议等进行漏洞分析&#xff0c;识别潜在的安全漏洞和弱点。这可能涉及对设备的渗透测…

佛教祭拜小程序-寺庙小程序-纪念馆小程序

大家好&#xff0c;我是程序员小孟。 现在有很多的产品或者工具都开始信息话了&#xff0c;寺庙或者佛教也需要小程序吗&#xff1f; 当然了&#xff01; 前面我们还开发了很多寺庙相关的小程序&#xff0c;都有相关的介绍&#xff1a; 1,优质的寺庙小程序-H5寺庙网页 今天…

jmeter常用的断言

包括&#xff08;Contains&#xff09;&#xff1a;响应内容包括需要匹配的内容即代表响应成功&#xff0c;支持正则表达式 匹配&#xff08;Matches&#xff09;&#xff1a;响应内容要完全匹配需要匹配的内容即代表响应成功&#xff0c;大小写不敏感&#xff0c;支持正则表达…

2. redis配置文件解析

redis配置文件解析 一、redis配置文件1、监听地址2、监听端口3、redis接收请求的队列长度3.1 修改系统参数/内核参数 4、客户端空闲的超时时间5、指定redis的pid文件6、定义错误日志7、定义数据库的数量8、定义持久化存储9、设置redis密码10、redis并发连接11、最大内存策略 二…

python的一种集成开发工具:PyCharm开发工具

一. 简介 本文简单了解两种 python语言所使用的 集成开发环境&#xff1a; PyCharm、vscode。 python语言学习中&#xff0c;可以任意选中这两个集成开发环境的一种就可以。本文先来简单学习 PyCharm开发工具安装与使用。 二. python的一种集成开发工具&#xff1a;PyChar…

Warning:成交前,永远相信意外即将发生

作为一名首次次创业者&#xff0c;随着创业进入深层次阶段&#xff0c;越来越感觉到&#xff1a;创业是一条不归路&#xff0c;因为路上不止有惊喜&#xff0c;还有风尘。创业之前我认为世界是“天圆地方”的&#xff0c; 创业后你猜我怎么看这个世界的&#xff1f; 创业前我一…

ARM32开发——串口库封装(初级)

&#x1f3ac; 秋野酱&#xff1a;《个人主页》 &#x1f525; 个人专栏:《Java专栏》《Python专栏》 ⛺️心若有所向往,何惧道阻且长 文章目录 开发流程分组创建 接口定义完整代码 开发流程 在文件系统中&#xff0c;创建库目录Library在keil工程中&#xff0c;创建分组管理…

健身日记之倒立俯卧撑学习——起始日2024.6.4

文章目录 前言 自我介绍 昔日计划 新目标计划 瓶颈突破尝试 参考视频及文章 前言 有轻微健身基础&#xff0c;正式接触街健五大神技&#xff0c;立志在两年内解锁全部&#xff0c;将有机会的进行日常训练和目标肌群锻炼&#xff0c;这里向大家展示我的计划和安排&#xf…

【C++练级之路】【Lv.24】异常

快乐的流畅&#xff1a;个人主页 个人专栏&#xff1a;《算法神殿》《数据结构世界》《进击的C》 远方有一堆篝火&#xff0c;在为久候之人燃烧&#xff01; 文章目录 引言一、异常的概念及定义1.1 异常的概念1.2 异常的定义 二、异常的使用2.1 异常的栈展开匹配2.2 异常的重新…

Django使用正则表达式

本书1-7章样章及配套资源下载链接: https://pan.baidu.com/s/1OGmhHxEMf2ZdozkUnDkAkA?pwdnanc 源码、PPT课件、教学视频等&#xff0c;可以从前言给出的下载信息下载&#xff0c;大家可以评估一下。 在Django框架的新版本&#xff08;v2.0 &#xff09;中&#xff0c;URLc…

电脑开机之后要很久才能进入系统?进入WinPE也是卡顿半天?

前言 小白最近接到了一张很奇怪的电脑维修单&#xff0c;客户说他的工作室电脑开机特别慢&#xff0c;开机之后特别卡顿&#xff0c;在使用的时候也会一卡一卡的。 这事情开始看很简单&#xff1a;估计就是电脑还是机械硬盘&#xff0c;所以开机很慢又卡顿。所以应该是把机械…

一键开启:盲盒小程序里的梦幻奇遇

在繁忙的都市生活中&#xff0c;每个人心中都藏着一个关于奇遇的梦想。如今&#xff0c;我们为您精心打造了一款盲盒小程序——“梦幻奇遇”&#xff0c;只需一键开启&#xff0c;就能带您走进一个充满无限惊喜和梦幻色彩的奇幻世界。 一、神秘盲盒&#xff0c;惊喜连连 “梦幻…

SolidWorks价格与其它CAD软件相比:为什么选择SolidWorks更划算

在CAD软件的浩瀚星海中&#xff0c;SolidWorks如同一颗璀璨的明星&#xff0c;以其卓越的性能、广泛的适用性和合理的定价策略&#xff0c;赢得了全球众多工程师和设计师的青睐。亿达四方&#xff0c;作为官方授权的SolidWorks代理商&#xff0c;今天将带您深入价格与功能的比较…