闸北做网站公司/互动营销经典案例
结构体+内存对齐+实现位段
- 一.结构体
- 1.结构体的声明
- 2.结构体变量成员访问操作符
- 3.结构体传参
- 4.匿名结构体
- 5.结构的自引用
- 二.结构体内存对齐
- 1.对齐规则
- 2.为什么存在内存对齐?
- 3.修改默认对齐数
- 三.结构体实现位段
- 1.什么是位段
- 2.位段的内存分配
- 3.位段的跨平台问题
- 4.位段的应用
- 5.位段使用的注意事项
前言:
- 学习了数组后发现数组中的元素只能是相同类型的变量,那么有没有可以存放不同类型的变量呢?
- 结构体:一些值的集合,这些值称为成员变量,结构体的每个成员可以是不同类型的变量。
一.结构体
1.结构体的声明
struct tag
{member-list;//结构体成员列表
}variable-list;//结构体变量列表
例如:描述一个人
struct Person {int age;//年龄char name[50];//姓名float height;//身高
};//封号不能丢
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.结构体传参
- 传结构体。
- 传结构体的地址。
#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;
}
思考:我们发现二者都可以成功访问结构体成员,那二者有什么区别呢?
- 传递结构体时:其实函数内部创建了一个临时结构体变量存放传入的结构体,当结构体很大时会额外占用空间不划算。(本质上是值传递)。
- 传递结构体地址时:只需创建4个字节结构体指针变量,通过其来访问结构体成员,可以大大节省空间。(本质上是地址/指针传递)。
- 推荐传递结构体地址。
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.对齐规则
- 结构体的第一个成员对齐到和结构体变量起始位置偏移量为 0 的地址处。
偏移量:该成员变量的地址距离结构体地址的字节数(计算偏移量的函数:offsetof)。 - 其他成员变量要对齐到某个数字(对齐数)的整数倍的地址处。
对齐数 = 编译器默认的对齐数与该成员变量大小的较小值。
在VS 中默认的对齐数值为 8 。
Linux中gcc编译器没有默认对齐数,对齐数就是成员自身的大小。 - 结构体总大小为最大对齐数(结构体中每个成员变量都有⼀个对齐数,所有对齐数中最大的)的整数倍。
- 如果嵌套了结构体的情况,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。
offsetof宏:计算结构体成员相较于结构体变量起始位置的偏移量,头文件stddef.h
例如:计算结构体大小的代码。
#include<stdio.h>
#include<stddef.h>
struct S1
{char c1;//自身大小1,默认对齐数8,对齐数1char c2;//自身大小1,默认对齐数8,对齐数1int n;//自身大小4,默认对齐数8,对齐数4
};
struct S2
{char c1;//自身大小1,默认对齐数8,对齐数1int n;//自身大小4,默认对齐数8,对齐数4char c2;//自身大小1,默认对齐数8,对齐数1
};
int main()
{printf("%zd\n", offsetof(struct S1, c1));//0printf("%zd\n", offsetof(struct S1, c2));//1printf("%zd\n", offsetof(struct S1, n));//4printf("%zd\n", sizeof(struct S1));//8printf("%zd\n", offsetof(struct S2, c1));//0printf("%zd\n", offsetof(struct S2, n));//4printf("%zd\n", offsetof(struct S2, c2));//8printf("%zd\n", sizeof(struct S2));//12return 0;
}
练习:
#include<stdio.h>
struct S1
{double d;//自身大小8,默认对齐数8,对齐数8char c;//自身大小1,默认对齐数8,对齐数1int i;//自身大小4,默认对齐数8,对齐数4
};
struct S2
{char c1;//自身大小1,默认对齐数8,对齐数1struct S1 s1;//自身大小16,默认对齐数8,对齐数8//如果嵌套了结构体的情况,嵌套的结构体成员对齐到《自己的成员中最大对齐数的整数倍处(d的对齐数的整数倍处)》,//结构体的整体大小就是所有最大对齐数(含嵌套结构体中成员的对齐数)的整数倍。double d;//自身大小8,默认对齐数8,对齐数8
};
int main()
{printf("%zd\n", sizeof(struct S1));//16printf("%zd\n", sizeof(struct S2));//32return 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));//12printf("%zd\n", sizeof(struct S2));//8//S1 和 S2 类型的成员⼀模⼀样,但是 S1 和 S2 所占空间的大小有了⼀些区别。return 0;
}
总结:让占用空间小的成员尽量集中在⼀起。
3.修改默认对齐数
- VS上默认对齐数为8。
- #pragma pack(一般为2^n) 这个预处理指令,可以改变编译器的默认对齐数。
- 例如#pragma pack(1),#pragma pack(2),#pragma pack(4)。
- #pragma pack() == #pragma pack(8)。
#include<stdio.h>
#pragma pack(1)//修改默认对齐数变成1
struct S
{char c1;//自身大小1,默认对齐数1,对齐数1int i;//自身大小4,默认对齐数1,对齐数1char c2;//自身大小1,默认对齐数1,对齐数1
};
#pragma pack()//将默认对齐数修改为8
int main()
{printf("%zd\n", sizeof(struct S));//6return 0;
}
三.结构体实现位段
- 结构体有实现位段的功能。
1.什么是位段
位段的声明和结构是类似的,有两个不同:
- 位段的成员必须是 int、unsigned int 或 signed int ,在C99中位段成员的类型也可以选择其他类型。
- 位段的成员名后边有一个冒号和一个数字。
- 位段中的位:二进制的位。
位段与结构体语法上的区别,代码如下:
#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.位段的内存分配
- 位段的成员可以是 int unsigned int signed int 或者是 char 等类型。
- 位段的空间上是按照需要以4个字节( int )或者1个字节( char )的方式来开辟的。
- 位段涉及很多不确定因素,位段是不跨平台的,注重可移植的程序应该避免使用位段。
#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.位段的跨平台问题
- int位段被当成有符号数还是无符号数是不确定的。
- 位段中最大位的数目不能确定。(16位机器最大16(sizeof(int)==2),32位机器最大32(sizeof(int)==4),写成27,在16位机器会出问题)。
- 位段中的成员在内存中从左向右分配,还是从右向左分配,标准尚未定义。
- 当⼀个结构包含两个位段,第⼆个位段成员比较大,无法容纳于第⼀个位段剩余的位时,是舍弃剩余的位还是利用,这是不确定的。
总结:跟结构体相比,位段可以达到同样的效果,并且可以很好的节省空间,但是有跨平台的问题存在。
4.位段的应用
- 下图是网络协议中,IP数据报的格式,我们可以看到其中很多的属性只需要几个bit位就能描述,这里使用位段,能够实现想要的效果,也节省了空间,这样网络传输的数据报大小也会较小⼀些,对网络的畅通是有帮助的。
- 在网络中发送数据的时候,需要进行数据的封装,例如:加上源地址与目的地址。(计算机网络中的网络层协议——> IP协议)。
- 为了避免网络拥堵,相办法节省空间,使用的就是位段。
5.位段使用的注意事项
- 位段的几个成员共有同⼀个字节,这样有些成员的起始位置并不是某个字节的起始位置,那么这些位置处是没有地址的。内存中每个字节分配一个地址,一个字节内部的bit位是没有地址的。
- 所以不能对位段的成员使用&操作符,这样就不能使用 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;
}
创作不易,如果能帮到你的话能赏个三连吗?感谢啦!!!
相关文章:

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

0109__strip(1) command
strip(1) command_linux strip-CSDN博客...

英码科技推出鸿蒙边缘计算盒子:提升国产化水平,增强AI应用效能,保障数据安全
当前,随着国产化替代趋势的加强,鸿蒙系统Harmony OS也日趋成熟和完善,各行各业都在积极拥抱鸿蒙;那么,边缘计算要加快实现全面国产化,基于鸿蒙系统开发AI应用势在必行。 关于鸿蒙系统及其优势 鸿蒙系统是华…...

从军事角度理解“战略与战术”
战略与战术,均源于军事术语。 战略(Strategy),源自希腊语词汇“strategos(将军)”和“strategia(军事指挥部,即将军的办公室和技能)”。指的是指挥全局性作战规划的谋略…...

最短路径——迪杰斯特拉与弗洛伊德算法
一.迪杰斯特拉算法 首先对于最短路径来说:从vi-vj的最短路径,不用非要经过所有的顶点,只需要找到路径最短的路径即可; 那么迪杰斯特拉的算法:其实也就与最小生成树的思想类似,找到较小的,然后…...
6.7.11 一种新的迁移学习方法可提高乳房 X 线摄影筛查中乳腺癌的诊断率
分割是一种将图像分割成离散区域的技术,以便将感兴趣的对象与周围环境分开。为了制定治疗计划,分割可以帮助医生测量乳房中的组织量。 二元分类问题的目的是将输入数据分为两组互斥的数据。在这种情况下,训练数据根据要解决的问题以二进制格…...

【Proteus8.16】Proteus8.16.SP3.exe的安装包,安装方法
下载: 链接:https://pan.baidu.com/s/14ZlETF7g4Owh8djLaHwBOw?pwd2bo3 提取码:2bo3 管理员打开proteus8.16.SP3.exe一路装就行了,许可证选Licence2.lxk,点安装后关闭,然后继续装完。 然后打开Patch-Proteus-8.16-…...

17、matlab实现均值滤波、中值滤波、Butterworth滤波和线性相位FIR滤波
1、创建信号 1)创建正余弦信号、噪声信号和混合信号 原始正余弦信号公式:Signal1 sin(2*pi*20* t) sin(2*pi*40* t) sin(2*pi*60* t) 高斯分布的白噪声:NoiseGauss [randn(1,2000)] 均匀分布的白噪声:[rand(1,2000)] 正余弦…...

【Autopilot】没有自动添加本地管理员的问题处理
【问题】某公司选用了D记的笔记本电脑,约定出厂就预配置好Autopilot,当时向D记提供了三个信息: 1. M365的租户ID 2. 公司域名信息 3. Group Tag (某公司为跨国公司,通过Group Tag来区分国家,比如CHN-中国,L…...

【C#学习笔记】属性和字段
文章目录 前言属性和字段的区别字段访问修饰符和关键字定义变量类型的定义变量命名变量的赋值 属性 不同的使用情况 前言 最近在工作的过程中常常会觉得自己在程序设计方面的能力还是有欠缺。例如一直对于变量的声明感到不足,在工作中为了图方便总是直接public定义…...

最佳实践的实践 - API 不应将 HTTP 重定向到 HTTPS
原文:jviide - 2024.05.23 TL;DR: 与其将 API 调用从 HTTP 重定向到 HTTPS,不如让失败显而易见。要么完全禁用 HTTP 接口,要么返回明确的 HTTP 错误响应,并撤销通过未加密连接发送的 API 密钥。遗憾的是,许多知名的 A…...

四种跨域解决方案
文章目录 1.引出跨域1.基本介绍2.具体演示1.启动之前学习过的springboot-furn项目2.浏览器直接访问 [localhost:8081/furns](http://localhost:8081/furns) 可以显示信息3.启动前端项目,取消请求拦截器,这样设置,就会出现跨域4.跨域原因 2.跨…...

移动端投屏到大屏幕的操作详解
如果你懒得折腾电脑、电视或其他大屏设备上的影视软件安装及配置,可以选择直接在手机端上将影片投屏到电脑、电视或其他大屏设备上,这里给大家分享三种手机投屏的方法。 系统自带的投屏功能 不管是安卓、鸿蒙还是苹果操作系统,都自带了无线…...

【环境搭建】3.阿里云ECS服务器 安装Redis
在阿里云的 Alibaba Cloud Linux 3.2104 LTS 64位系统上安装 Redis 可以通过以下步骤完成: 1.更新系统软件包: 首先,更新系统软件包以确保所有软件包都是最新的: sudo yum update -y2.安装编译工具和依赖项: Redis…...

动态语言的开源编译器汇总
对于动态语言而言,我们通常不会使用传统意义上的“编译器”,因为动态语言往往是在运行时解释执行的,或者被转换为中间形式(如字节码),再由虚拟机执行。不过,为了性能考虑,现代动态语…...

Linux防火墙配置001
Linux防火墙主要用于控制网络流量,保护系统安全。在Linux中,有几种不同的防火墙管理工具,其中最常见的是iptables和firewalld。本章主要讲述如何关闭防火墙。 操作系统: CentOS Stream 9 操作步骤: 关闭防火墙&…...

Tomcat概述及部署
目录 一.Tomcat概述 1.介绍 2.使用场景 3.组件构成 4.组件结构 5.请求过程 二.Tomcat部署 1.关闭防火墙 2.下载安装JDK 3.安装启动tomcat 4.部署虚拟主机 4.1.创建 xy101 和 xy102 项目目录和文件 4.2.修改 Tomcat 主配置文件 server.xml 一.Tomcat概述 1.介绍 …...

[Vue3:Vite构建项目]:安装router实现登录页面路由跳转
文章目录 一:前置依赖查看依赖安装vite npm create vitelatest sys-instruction-0607 --template vue-ts安装路由:npm install vue-router4安装elementUI:npm install element-plus --save 二:配置文件:viewsÿ…...

概率论与数理统计,重要知识点——全部公式总结
二、一维随机变量及其分布 五个分布参考另外一篇文章 四、随机变量的数字特征 大数定理以及中心极限定理 六、数理统计...

Spring系列-SpringMvc父子容器启动原理解析
1、Spring整合SpringMVC 特性: 说到Spring整合SpringMVC唯一的体现就是父子容器: 通常我们会设置父容器(Spring)管理Service、Dao层的Bean, 子容器(SpringMVC)管理Controller的Bean .子容器可以访问父容器的Bean, 父容器无法访…...

[ssi-uploader插件]解决如何接收服务器返回数据+修改参数名称
前言 ssi-uploader是一款非常好用的多文件上传插件,源码是开源的,在github上面即可下载: https://github.com/ssbeefeater/ssi-uploader 但是源码有些微小的不足,今天我们解决两点问题: 上传文件完成后,…...

InfiniGate自研网关实现思路七
25.网关Nginx负载模型配置 通过模拟多个HTTP服务配置到 Nginx 做负载均衡,以学习API网关负载的配置和使用 API 网关是用于支撑分布式 RPC 接口协议转换提供 HTTP 调用的一套服务,那么 API 网关系统就需要可横向扩展来满足系统的吞吐量诉求。所以这里需…...

277 基于MATLAB GUI火灾检测系统
基于MATLAB GUI火灾检测系统,可以实现图片和视频的火苗检测。火焰识别的三个特征:1个颜色特征,2个几何特征颜色特征:HSV颜色空间下,对三个通道值进行阈值滤波,几何特征1:长宽比,几何…...

【西瓜书】4.决策树
1 递归返回情况 (1)结点包含样本全为同一类别 (2)属性集为空,没有属性可供划分了 或 有属性,但是在属性上划分的结果都一样 (3)结点为空结点 **结束时判定该结点的类别遵循如下规则&…...

区块链--Ubuntu上搭建以太坊私有链
1、搭建私链所需环境 操作系统:ubuntu16.04,开虚拟机的话要至少4G,否则会影响测试挖矿时的速度 软件: geth客户端 Mist和Ethereum Wallet:Releases ethereum/mist GitHub 2、安装geth客户端 sudo apt-get update …...

菜品信息分页查询——后端SpringBoot
1.分页查询的逻辑: 页面发送ajax请求,将分页查询参数(page,pageSize, name)提交到服务端,获取分页数据; 页面发送请求,请求服务端进行图片下载,用于页面图片展示。 开发菜品信息分页查询功能&a…...

利用GPT和PlantUML快速生成UML图用于设计
在软件开发中,设计阶段可是关键的一步。UML(统一建模语言)图能帮我们更清晰地理解和规划系统结构,但手动画UML图有时会很费时费力。好消息是,通过结合使用ChatGPT和PlantUML,我们可以高效地生成UML图&#…...

web-上传项目文件夹到Git远程仓库
Git初识 概念:一个免费开源,分布式的代码版本控制系统,帮助开发团队维护代码 作用:记录代码内容,切换代码版本,多人开发时高效合并代码内容 检验成功 打开bash终端(git专用)命令…...

使用OpenPCDet训练与测试Transformer模型:如何加载自己的数据集
引言 Transformer架构因其强大的序列处理能力和长距离依赖捕捉能力,在自然语言处理领域取得了巨大成功。近年来,这一架构也被引入3D物体检测领域,如Voxel Transformer等,显著提升了模型在复杂场景下的检测性能。OpenPCDet整合了多…...

四舍五入问题
单纯输出四舍五入可以用 printf("%.nf",num); 但是这个方法有时候会出错 代表输出n位四舍五入小数 而将数四舍五入赋值给变量可以用round()函数 a round(num); 表示将num四舍五入赋值给a 但是这么些只能转换位四舍五入的整数 可以改…...