【C语言】结构体详解 (二) 内存函数、结构体传参

目录

1、 结构体的内存对齐

1.1、对齐规则

1.2、练习1、练习2(演示对齐规则1、2、3、4)

2、为什么存在内存对齐

2.1、平台原因(移植原因)

2.2、性能原因

2.3、那么如何即满足对齐,又要节省空间呢?

3、修改默认对齐数

4、结构体传参

4.1、将结构体传到函数print中

4.2、将地址传到函数print中

4.3、区别

5、结构体实现位段

5.1、什么是位段

5.2、位段的内存分配

5.3、注意事项

6、谢谢观看


上一篇博客,写了结构体变量的创建、初始化和声明等内容,今天的这篇博客来带大家深入理解结构体的知识点。希望大家多多支持。 

正文 

1、 结构体的内存对齐

首先,抛一个问题:结构体的大小如何计算?

要知道这个题的答案,首先要了解结构体内存对齐

1.1、对齐规则

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

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

3、结构体总大小为最大对齐数的整数倍

4、如果嵌套了结构体,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数中的最大值的整数倍

偏移量:字节与结构体变量开始存放位置之间相偏移的值

对齐数:编译器默认的一个对齐数与该成员变量大小相比  二者之中取。较小值 

       VS中默认的一个对齐数是  8

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

最大对齐数:结构体中每个成员变量都有一个对齐数,所有对齐数中最大的数

1.2、练习1、练习2(演示对齐规则1、2、3、4)

求结构体的大小

练习1、

(演示对齐规则1、2 、3)

找对齐数: 

对齐数: 编译器默认的一个对齐数与该成员变量大小 相比 二者之中取较小值。

c1  的对齐数是 1

i  的对齐数是4

c2  的对齐数是1

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

如下图:第一个成员变量c1 放在偏移量为0的位置

对齐规则2:其他成员变量要对齐到其对齐数的整数倍的地址处

成员  i  的对齐数是 4,i 从偏移量为4的倍数的位置开始存放,按本题即从偏移量为4的位置开始,向后存放4个字节。 

 成员  c2  的对齐数是 1, c2  从偏移量为1的倍数的位置开始存放,按本题即从偏移量为8的位置开始,向后存放1个字节。

对齐规则3: 结构体总大小为最大对齐数的整数倍

结构体中三个成员的对齐数分别为  1、4、1,则最大对齐数是 4

那么结构体总大小为 4 的整数倍

由上图,三个成员已经占了9个字节的空间,所以不能少于4的2倍为8

则结构体总大小为  4*3=12,  4的3倍

练习2、

(演示对齐规则4)

对齐规则4: 如果嵌套了结构体,嵌套的结构体成员对齐到自己的成员中最大对齐数的整数倍处,结构体的整体大小就是所有最大对齐数中的最大值的整数倍

对于 struct S2   (内嵌结构体),其结构体总大小为2*8=16

内嵌结构体的最大对齐数是  8 

内嵌结构体的最大对齐数是  8 ,则在结构体S3中该结构体的对齐数为8 ,大小为16

由上图,该结构体的大小为  4*8=32 

2、为什么存在内存对齐

2.1、平台原因(移植原因)

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

举例说明:有些平台上结构体成员中 int类型的数据只能存在4的倍数的内存中,此时就需要有内存对齐。

2.2、性能原因

数据结构(特别是栈)应该尽可能的在自然边界上对齐。原因:为了访问未对齐的内存 ,处理器需要做两次内存访问;而对齐的内存访问只需要一次。

例如:

在32为平台下,一次访问4个字节,成员i  在对齐的情况下能被一次读完。

不对齐的情况下(按顺序存放)

所以说,内存对齐损耗了空间,但节省了时间,结构体的内存对齐是拿空间来换取时间的做法。 

2.3、那么如何即满足对齐,又要节省空间呢?

请看下面的例子:(两个结构体中只是更改了成员的顺序)

struct S1中 两个占空间小的char 类型的成员分散排列。

而 struct S2中 两个占空间小的char 类型的成员集中在一起排列。 

所以要即满足对齐,又要节省空间的方法是:让占用空间小的成员尽量集中在一起

3、修改默认对齐数

使用 #pragma 这个预处理命令,可以修改编译器的默认对齐数。

具体使用: 

设置默认对齐数为1,相当于不对齐的情况,所占字节是所有成员的字节大小。

结构体在对齐方式不合适的时候,我们可以自己更改默认对齐数。 

4、结构体传参

结构体传参可以传结构体,也可以传地址。但我们首选传地址。

4.1、将结构体传到函数print中

4.2、将地址传到函数print中

4.3、区别

传结构体:在传结构体时需要创建临时结构体来储存,如果结构体中有成员占内存过大,会在传递时产生时间和空间的巨大开销。

正经解释:

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

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

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

5、结构体实现位段

结构体具有实现位段的能力。

5.1、什么是位段

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

基本形式:位段成员名后面有一个冒号和一个数字。数字代表该成员所占的bit位数。

这里的A就是位段类型。

5.2、位段的内存分配

  • 位段的空间上是按照以4个字节或1个字节的方式来开辟的。
  • 位段涉及很多的不确定因素,是不能跨平台的。 

 详细开辟方式如结构体。

5.3、注意事项

 不能对位段的成员使用&操作符,这样就不能使用scanf直接给位段的成员输入值,只能是先输入放在一个变量中,然后赋值给位段成员。

如下:

6、谢谢观看

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

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

相关文章

【Vue3源码学习】— CH2.5 reactiveEffect.ts:Vue 3响应式系统的核心

reactiveEffect.ts:Vue 3响应式系统的核心 1. 什么是 reactiveEffect?2. 核心机制2.1 依赖收集(Track)2.2 触发更新(Trigger)2.3 效果范围(effectScope) 3. 源码解析 —— track3.1 …

卷积神经网络(CNN)——基础知识整理

文章目录 1、卷积神经网络 2、图片格式 3、图片卷积运算 4、Kernel 与 Feature Map 5、padding/边缘填充 6、Stride/步长 7、pooling/池化 8、shape 9、epoch、batch、Batch Size、step 10、神经网络 11、激活函数 1、卷积神经网络 既然叫卷积神经网络,这里面首先是…

4、Cocos Creator 动画系统

目录 1、Clip 参数 2、动画编辑器 3、基本操作 更改时间轴缩放比例 移动显示区域 更改当前选中的时间轴节点 播放 / 暂停动画 修改 clip 属性 快捷键 4、模拟实验 5、动画事件 6、注意事项 参考 Animation 组件是节点上的一个组件。Clip 动画剪辑就是一份动画的声…

67、yolov8目标检测和旋转目标检测算法部署Atlas 200I DK A2开发板上

基本思想:需求部署yolov8目标检测和旋转目标检测算法部署atlas 200dk 开发板上 一、转换模型 链接: https://pan.baidu.com/s/1hJPX2QvybI4AGgeJKO6QgQ?pwdq2s5 提取码: q2s5 from ultralytics import YOLO# Load a model model YOLO("yolov8s.yaml")…

物流监控升级,百递云·API开放平台助力某电商平台实现高效物流管理

不论是电商平台自身还是消费者,都有着对物流监控的强烈需求。 消费者下单后be like: 每十分钟看一次快递轨迹 放心,电商平台也一样关心物流状态!怎样第一时间向用户传递物流状态? 怎么减少重复的提问和投诉?怎样监管…

Collection与数据结构 顺序表与ArrayList

1. 线性表 线性表(linear list)是n个具有相同特性的数据元素的有限序列。 线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列… 线性表在逻辑上是线性结构,也就说是连续的一条直线。但是在…

【Docker】搭建强大的Nginx可视化配置工具 - nginxWebUI

【Docker】搭建强大的Nginx可视化配置工具 - nginxWebUI 前言 本教程基于绿联的NAS设备DX4600 Pro的docker功能进行搭建。 简介 NginxWebUI是一个基于Java的,专门用来管理Nginx的图形界面工具。它是开源的,使用相对简单且功能全面。 使用NginxWebUI…

接劳巴,拔掉KL15,MCU重启。不接劳巴,拔掉KL15,MCU正常下电

最近遇到一个神奇的现象,在调试一个单KL15的项目,发现接着劳特巴赫调试器,然后拔掉KL15,软件进入了重启Reset函数,没有进入期望的下电SwitchOff函数。 而不接劳特巴赫,拔掉KL15,观测电流&#…

Qt实现Kermit协议

1 概述 Kermit文件运输协议提供了一条从大型计算机下载文件到微机的途径。它已被用于进行公用数据传输。 其特性如下: Kermit文件运输协议是一个半双工的通信协议。它支持7位ASCII字符。数据以可多达96字节长度的可变长度的分组形式传输。对每个被传送分组需要一个确认。Kerm…

python-判断列表字典循环

比较运算符 不等于 ! if 布尔值: [执行语句-真实执行] else: [执行语句] mood_index int(input("对象今天的心情指数的是:")) if mood_index > 60:print("恭喜,今晚应该可以带游戏,去吧")…

2024年水电站大坝安全监测工作提升要点

根据《水电站大坝运行安全监督管理规定》(国家发改委令第23号)和《水电站大坝运行安全信息报送办法》(国能安全〔2016〕261号)的相关规定、要求,电力企业应当在汛期向我中心报送每日大坝汛情。近期,全国各地…

【机器学习】深度解析KNN算法

深度解析KNN算法 KNN(K-最近邻)算法是机器学习中一种基本且广泛应用的算法,它的实现简单直观,应用范围广泛,从图像识别到推荐系统都有其身影。然而,随着数据量的增长,KNN算法面临着严峻的效率挑…

[yolox]ubuntu上部署yolox的ncnn模型

首先转换pytorch->onnx->param模型,这个过程可以查资料步骤有点多,参考blog.51cto.com/u_15660370/6408303,这里重点讲解转换后部署。 测试环境: ubuntu18.04 opencv3.4.4(编译过程省略,参考我其他博客) 安装…

【智能家居入门3】(MQTT服务器、MQTT协议、微信小程序、STM32)

前面已经写了三篇博客关于智能家居的,服务器全都是使用ONENET中国移动,他最大的优点就是作为数据收发的中转站是免费的。本篇使用专门适配MQTT协议的MQTT服务器,有公用的,也可以自己搭建(应该要钱)&#xf…

常见的数学方法

Math类表示数学类,其中的数学方法都被定义成为static形式,所以可以直接通过Math类的类名调用某个数学方法。语法格式: Math.xxx(参数); 例题 输入n个整数a1,a2,a3,......an,求这n个数的最大值max,最小值min&#xff0…

算法之并查集

并查集(Union-find Data Structure)是一种树型的数据结构。它的特点是由子结点找到父亲结点,用于处理一些不交集(Disjoint Sets)的合并及查询问题。 Find:确定元素属于哪一个子集。它可以被用来确定两个元…

【御控物联】 IOT异构数据JSON转化(场景案例一)

文章目录 前言技术资料 前言 随着物联网、大数据、智能制造技术的不断发展,越来越多的企业正在进行工厂的智能化转型升级。转型升级第一步往往是设备的智能化改造,助力设备数据快速上云,实现设备数据共享和场景互联。然而,在生产…

【蓝桥杯】填空题技巧|巧用编译器|用Python处理大数和字符|心算手数|思维题

目录 一、填空题 1.巧用编译器 2.巧用Excel 3. 用Python处理大数 4.用Python处理字符 5.心算手数 二、思维题 推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。【点击跳转到网站】 一、填空题 …

蓝桥杯day14刷题日记

P8707 [蓝桥杯 2020 省 AB1] 走方格 思路&#xff1a;很典型的动态规划问题&#xff0c;对于偶数格特判&#xff0c;其他的正常遍历一遍&#xff0c;现在所处的格子的方案数等于左边的格子的方案数加上上面格子的方案数之和 #include <iostream> using namespace std; …

蓝桥杯物联网竞赛_STM32L071_13_定时器

CubeMx配置LPTIM: counts internal clock events 计数内部时钟事件 prescaler 预分频器 updata end of period 更新期末 kil5配置&#xff1a; 中断回调函数完善一下&#xff1a; void HAL_LPTIM_AutoReloadMatchCallback(LPTIM_HandleTypeDef *hlptim){if(cnt ! 10) cnt…