C语言-结构体尺寸

news2025/1/13 17:33:06

CPU字长

字长的概念指的是处理器在一条指令中的数据处理能力,当然这个能力还需要搭配操作系统的设定,比如常见的32位系统、64位系统,指的是在此系统环境下,处理器一次存储处理的数据可以达32位或64位。

地址对齐

当计算机系统的CPU字长确定后,就意味着每次CPU从内存中读取或写入数据时都会以固定的边界进行操作。以32位系统为例,这意味着CPU每次操作都会以4字节(32位)为单位。

如果所需数据恰好在这个4字节的边界上,我们称其为地址对齐的。这样的数据读取或写入会更高效,因为CPU可以直接按照这个边界进行操作。

而如果数据跨越了这个边界,使用了超过所需单元的字节,我们称其为地址未对齐的。这会导致额外的处理工作,可能会降低效率,并且在某些系统上可能会导致错误或异常。

地址对齐是一种优化技术,有助于提高系统的性能和稳定性。

 地址未对齐的情形                                                                地址已对齐的情形

从图中可以明显看出,数据本身占据了8个字节,在地址未对齐的情况下,CPU需要分3次才能完整地存取完这个数据,但是在地址对齐的情况下,CPU可以分2次就能完整地存取这个数据。

总结:
如果一个数据满足以最小单元数存放在内存中,则称它地址是对齐的,否则是未对齐的。地址对齐的含义用大白话说就是1个单元能塞得下的就不用2个;2个单元能塞得下的就不用3个。
如果发生数据地址未对齐的情况,有些系统会直接罢工,有些系统则降低性能。

普通变量的m值

以32位系统为例,由于CPU存取数据总是以4字节为单元,因此对于一个尺寸固定的数据而言,当它的地址满足某个数的整数倍时,就可以保证地址对齐。这个数就被称为变量的m值。
根据具体系统的字长,和数据本身的尺寸,m值是可以很简单计算出来的。

  • 举例 ( 注意,变量的m值跟变量本身的尺寸有关,但它们是两个不同的概念)
char   c; // 由于c占1个字节,因此c不管放哪里地址都是对齐的,因此m=1
short  s; // 由于s占2个字节,因此s地址只要是偶数就是对齐的,因此m=2
int    i; // 由于i占4个字节,因此只要i地址满足4的倍数就是对齐的,因此m=4
double f; // 由于f占8个字节,因此只要f地址满足4的倍数就是对齐的,因此m=4

printf("%p\n", &c); // &c = 1*N,即:c的地址一定满足1的整数倍
printf("%p\n", &s); // &s = 2*N,即:s的地址一定满足2的整数倍
printf("%p\n", &i); // &i = 4*N,即:i的地址一定满足4的整数倍
printf("%p\n", &f); // &f = 4*N,即:f的地址一定满足4的整数倍
  • 手工干预变量的m值:
char c __attribute__((aligned(32))); // 将变量 c 的m值设置为32
  • 语法:
    • attribute 机制是GNU特定语法,属于C语言标准语法的扩展。
    • attribute 前后都是双下划线,aligned两边是双圆括号。
    • attribute 语句,出现在变量定义语句中的分号前面,变量标识符后面。
    • attribute 机制支持多种属性设置,其中 aligned 用来设置变量的 m 值属性。
    • 一个变量的 m 值只能提升,不能降低,且只能为正的2的n次幂。

结构体的M值

  • 概念:
    • 结构体的M值,取决于其成员的m值的最大值。即:M = max{m1, m2, m3, …};
    • 结构体的和地址和尺寸,都必须等于M值的整数倍。
  • 示例:
struct node
{
    short  a; // 尺寸=2,m值=2
    double b; // 尺寸=8,m值=4
    char   c; // 尺寸=1,m值=1
};

struct node n; // M值 = max{2, 4, 1} = 4;
  • 以上结构体成员存储分析:
    • 结构体的M值是4,这意味着结构体的地址和尺寸都必须是4的倍数。
    • 成员a的M值是2,但由于它是结构体的首元素,必须满足结构体的M值约束,所以a的地址必须是4的倍数。
    • 成员b的M值是4,所以在a和b之间需要填充2个字节的无效数据(通常是0),以保证b的地址是4的倍数。
    • 成员c的M值是1,它直接紧挨在b的后面,占据一个字节的空间。
    • 为了让结构体的尺寸满足4的倍数,c后面还需要填充3个字节的无效数据。
  • 以上结构体成员图解分析:

结构体成员布局

可移植性

可移植指的是相同的一段数据或者代码,在不同的平台中都可以成功运行。

  • 对于数据来说,有两方面可能会导致不可移植:
    • 数据尺寸发生变化
    • 数据位置发生变化
  • 第一个问题,是基本的数据类型在不同的系统所占据的字节数不同造成的,解决办法如下

使用可移植性数据类型来解决基本数据类型在不同系统中占据的字节数不同的问题。可移植性数据类型是指具有固定大小的数据类型,无论在何种平台上都具有相同的大小。在C语言中,可以使用<stdint.h>头文件中定义的标准整数类型来实现可移植性。这些类型包括int8_tint16_tint32_tint64_t等,它们分别表示有符号整数类型,且确保在任何平台上都有固定的字节数。 

示例:

#include <stdint.h>

int8_t   // 有符号8位整数
int16_t  // 有符号16位整数
int32_t  // 有符号32位整数
int64_t  // 有符号64位整数
  • 第二个问题,即数据位置的可移植性,可以通过使用特定的编译指令或者技术来解决

        解决这个问题的两种方法如下:

  1. 第一种方法是通过使用编译器提供的特性,固定每个成员的对齐方式,从而固定成员之间的间隔,确保结构体在不同平台上的布局一致。在这个例子中,通过使用`__attribute__((aligned(n)))`,指定每个成员的对齐方式,其中`n`为对齐值。这样,每个成员之间的间隔就是固定的。
    struct node
    {
        int8_t  a __attribute__((aligned(1))); // 将 m 值固定为1
        int64_t b __attribute__((aligned(8))); // 将 m 值固定为8
        int16_t c __attribute__((aligned(2))); // 将 m 值固定为2
    };
    
  2. 第二种方法是使用`__attribute__((packed))`属性,这会告诉编译器不要在结构体成员之间插入任何填充字节,从而让结构体的布局更加紧凑。这种方法会减少内存的浪费,但可能会降低访问效率,因为访问未对齐的数据可能会导致性能损失。

        

struct node
{
    int8_t  a;
    int64_t b;
    int16_t c;
} __attribute__((packed));

无论使用哪种方法,都可以确保结构体在不同平台上的布局一致,从而提高了数据的可移植性。

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

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

相关文章

智能酒精壁炉与酒店会客厅的氛围搭配

智能酒精壁炉在酒店会客厅的氛围搭配可以创造出舒适、温馨和现代的环境。以下是智能酒精壁炉与酒店会客厅氛围搭配的优势和建议&#xff1a; 提升装饰效果&#xff1a; 安装智能酒精壁炉可以显著提升会客厅的装饰效果。壁炉作为焦点装饰&#xff0c;增添了现代感和奢华感&…

STM32的端口引脚的复用功能及重映射功能解析

目录 STM32的端口引脚的复用功能及重映射功能解析 复用功能 复用功能的初始化 重映射功能 重映射功能的初始化 复用功能和重映射的区别 部分重映射与完全重映射 补充 STM32的端口引脚的复用功能及重映射功能解析 复用功能 首先、我们可以这样去理解stm32引脚的复用功能…

车道分割YOLOV8-SEG

车道分割YOLOV8-SEG&#xff0c;训练得到PT模型&#xff0c;然后转换成ONNX&#xff0c;OPENCV的DNN调用&#xff0c;支持C,PYTHON,ANDROID开发 车道分割YOLOV8-SEG

【C++ | 复合类型】结构体、共用体、枚举、引用

&#x1f601;博客主页&#x1f601;&#xff1a;&#x1f680;https://blog.csdn.net/wkd_007&#x1f680; &#x1f911;博客内容&#x1f911;&#xff1a;&#x1f36d;嵌入式开发、Linux、C语言、C、数据结构、音视频&#x1f36d; ⏰发布时间⏰&#xff1a; 本文未经允许…

AndroidStudio中虚拟机(AVD)无法启动,出现unable to locate adb错误

1.检查Android SDK Platform-Tools是否安装(个人是通过这个方法解决的) 首先通过File-Project Structure-Project SDK检查SDK有没有被选中 步骤&#xff1a;打开file -> settings &#xff0c;搜索SDK 之后点击"-",在点击Apply进行安装 2.可能是驱动的问题 电脑…

Web前端开发 小实训(二) 简易计算器

实训目的 学生能够使用函数完成简易计算器编写 操作步骤 1、请将加减乘除四个方法生成为以下函数&#xff0c;且有返回值 中文英语加法add减法subtract乘法multi除法division次幂pow()平方根sqrt() 提示&#xff1a; 除法中的除数不能为0&#xff01; 参考代码&#xff1…

OpenHarmony实战开发-动画概述

UI&#xff08;用户界面&#xff09;中包含开发者与设备进行交互时所看到的各种组件&#xff08;如时间、壁纸等&#xff09;。属性作为接口&#xff0c;用于控制组件的行为。例如&#xff0c;开发者可通过位置属性调整组件在屏幕上的位置。 属性值的变化&#xff0c;通常会引…

python作业 切片逆转

题目&#xff1a; &#xff08;反转显示一个整数&#xff09;编写下面的函数&#xff0c;反向显示一个整数。 列如&#xff1a;reserse(3456)。编写一个测试程序&#xff0c;提示用户输入一个整数&#xff0c;然后显示它的反向数。 第一步定义一个函数&#xff1a; def rev…

Docker容器部署overleaf

overleaf在线版限制很多&#xff0c;好在开源&#xff0c;准备在本地Docker部署&#xff0c;网上翻了翻&#xff0c;似乎本地部署并非易事&#xff0c;我也尝试了一下&#xff0c;发现直接使用docker-compose拉官方最新镜像部署的确问题很多&#xff0c;不过最终还是完美解决。…

前端学习<四>JavaScript——54-原型链

常见概念 构造函数 构造函数-扩展 原型规则和示例 原型链 instanceof 构造函数 任何一个函数都可以被 new&#xff0c;new 了之后&#xff0c;就成了构造方法。 如下&#xff1a; function Foo(name, age) {this.name name;this.age age;//retrun this; //默认有这…

系统思考—高效激励和沟通艺术

前两天结束建工集团的下属公司《高效激励和沟通艺术》的课程&#xff0c;不仅讨论了一些实用的管理技巧&#xff0c;更多的是从系统思考的视角&#xff0c;探讨了“结构如何影响行为”。这种方法让我们认识到系统力量的重要性&#xff0c;并且通过理解这些力量&#xff0c;我们…

武汉星起航:亚马逊助力中国卖家扬帆全球,共筑跨境电商新篇章

自1995年7月诞生以来&#xff0c;亚马逊以其前瞻性的视野和不懈的创新精神&#xff0c;迅速崛起为全球电商领域的领军者。总部设在美国西雅图的亚马逊&#xff0c;以其卓越的技术实力、丰富的商品资源和优质的服务体验&#xff0c;赢得了全球消费者的广泛认可。 随着全球化的加…

数据结构 - 队列 [动画+代码注释超详解],萌新轻松上手!!!

一. 队列的概念 队列是一种特殊的线性表&#xff0c;用于存储元素&#xff0c;并且按照先进先出(First In First Out)的顺序进行管理&#xff0c;这意味着最先加入队列的元素将会是最先从队列中被移除的元素 队列的原型&#xff1a;只允许在一端进行插入数据的操作&#xff0c…

ArcGIS Pro 和 Python — 分析全球主要城市中心的土地覆盖变化

第一步——设置工作环境 1–0. 地理数据库 在下载任何数据之前,我将创建几个地理数据库,在其中保存和存储所有数据以及我将创建的后续图层。将为我要分析的五个城市中的每一个创建一个地理数据库,并将其命名为: “Phoenix.gdb” “Singapore.gdb” “Berlin.gdb” “B…

入户厨房设计,220大平层现代风三室装修。福州中宅装饰,福州装修

设计亮点 220㎡的平层住宅需要在氛围、功能和储物空间方面进行质的提升&#xff0c;以满足三口之家的需求。原始结构包括五个卧室和两个客厅&#xff0c;客餐厅整体通透&#xff0c;但厨房稍显局促&#xff0c;主卧功能分区不够清晰。 入户厨房 设计亮点 这个充满现代氛围的入…

SpringBoot---------Hutool

第一步&#xff1a;引入依赖 <dependency><groupId>cn.hutool</groupId><artifactId>hutool-parent</artifactId><version>5.7.17</version></dependency> 第二步&#xff1a;各种用法 ①生成随机数 //生成验证码 String s …

29.Gateway网关的全局过滤器GlobalFilter

全局过滤器的作用也是处理一切进入网关的请求和微服务响应。 与GatewayFilter的作用一样(filters, default-filters) 区别 GatewayFilter通过配置定义&#xff0c;处理逻辑是固定的。 GlobalFilter的逻辑需要自己写代码实现&#xff0c;可以自定义。 exchange表示 请求上下…

EXCEL表格中的数字,为什么每次打开会自动变成日期?

一、典型现象 在工作中&#xff0c;有时会发现公司里的报表&#xff0c;经过多人多次的重复的使用和修改后&#xff0c;会出现这种情况&#xff1a; 1.在表格里按照需要输入数字&#xff0c;保存工作簿。 2.然而&#xff0c;再次打开工作簿&#xff0c;里面的数字变成日期&a…

【嵌入式AI开发】轻量级卷积神经网络MobileNetV2详解

前言:MobileNetV2网络先升维后降维,在降维时使用线性激活函数,带残差的Inverted bottleck模块,防止ReLU信息丢失。在图像分类、目标检测、语义分割等任务上实现了网络轻量化、速度和准确度的权衡。 回顾MobileNetV1的理论和MobileNetV2项目实战可查阅如下链接: 【嵌入式AI…

【前端开发基础知识快速入门】

前端开发基础知识&快速入门 一、VSCode 使用1.1 安装常用插件1.2 创建项目1.3 创建网页1.4 运行效果二、ES62.1 简介2.2 什么是 ECMAScript2.3 ES6 新特性2.3.1 let 声明变量2.3.2 const 声明常量(只读变量)2.3.3 解构表达式2.3.4 字符串扩展2.3.5 函数优化2.3.6 对象优化…