内存数据对齐(翻:Data alignment:straighten up and fly right)

2023-11-02 04:40

本文主要是介绍内存数据对齐(翻:Data alignment:straighten up and fly right),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

    1.内存访问粒度

     程序员习惯性把内存当做简单的字节数组,在c语言和它之后的语言中,char *代表“a block of memory”,甚至java有byte[]的类型去代表原始内存。程序员看到的内存如下图所示:


然而你的电脑处理器不会按照字节读或者写内存。相反,它访问内存是以2-,4-,8-,16-,32-字节为块。我们将处理器访问内存的大小叫做内存的访问粒度。处理器看到的内存如下图所示:

程序员认为的内存和现代处理器实际用内存工作的方式之间的不同,是本文探讨的有趣问题。

如果你不了解和解决数据对齐问题,以下情况都是可能发生的:

1.你的软件运行越来越慢

2.你的应用程序将被锁定

3.你的操作系统将会崩溃

4.你的软件将会产生错误结果



   2.内存对齐的基本知识

   为了更好地阐述对齐背后的原理,我们做一个task检测,了解内存怎么被处理器内存访问粒度影响。task很简单:首先从地址0开始读4个字节到处理器的寄存器。然后从地址1读4个字节到相同寄存器:


这符合天真的程序员所认为的内存工作模式:它要花四次内存访问从地址0开始读取,正如从地址1开始读取。现在看看如果处理器以2个字节为粒度将发生什么?


   当从地址0开始读取时,2字节粒度读取的处理器将花费1字节粒度读取的处理器一半的内存访问次数。因为每一个内存访问都会引起固定的开销,最小化访问的次数能够帮助提升性能。

   然而,从地址1开始读取会发生什么呢?因为地址不均匀地落在处理器的内存访问边界,处理器就有额外的工作要做。这样的地址就称为不对齐地址。因为地址1是不对齐的,2字节粒度的处理器就必须要执行额外的内存访问,减慢了操作。

   最后,我们将测试4字节内存访问粒度的处理器将发生什么?


   4字节内存访问粒度的处理器会一次性读取4个bytes,从一个对齐地址。同样值得注意的是,从一个不对齐地址读取将会加倍内存访问次数。接下来将具体解释处理器是怎样从一个不对齐地址进行内存读取操作的。

    

    Lazy 处理器

   当一个处理器要访问一个不对齐地址的时候,必须要执行一些技巧。回到刚刚使用4字节粒度的处理器从地址1开始在内存中读取4字节。你能够精确计算出什么需要被做:


处理器需要读取非对齐地址所在的第一块,然后从第一块中移除不想要的字节。然后它需要非对齐地址的第二块,移除它的某些信息,最终两个块的内容合并放置在寄存器中,这需要花费大量的工作。

一些处理器是愿意这么做的。

 原始的68000是两字节为粒度的处理器,它缺乏处理不对齐地址的电路。当面对不对齐的地址,处理器将会抛出异常。原始的MAC OS也不能很友好地解决这个异常,通常需要用户重启系统。

   在之后的680x0系列的处理器中,比如68020,解决了这个限制,执行必要的工作。这就解释了为什么一些老的软件可以运行在68020上,但在68000上却会崩溃。

   PowerPC采用了混合方式。每一种POWERPC处理器对不对齐的32位整型访问都有支持,同时,它一样对不对齐地址访问要付出一点性能补偿,但是这是很小的。

   在另一方面,现代PowerPC对于不对齐的64位浮点型访问缺乏硬件支持,当从内存中访问不对齐的浮点数,现代的PowerPC处理器会丢出异常同时操作系统会在软件层次执行对齐。在软件层次执行对齐是比在硬件层次执行对齐更慢的。


总结:

许多实际的计算机系统对基本类型数据在内存中存放的位置有限制,它们会要求这些数据的首地址的值是某个数k(通常它为4或8)的倍数,这就是所谓的内存对齐。






这篇关于内存数据对齐(翻:Data alignment:straighten up and fly right)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/328183

相关文章

MySQL中查询和展示LONGBLOB类型数据的技巧总结

《MySQL中查询和展示LONGBLOB类型数据的技巧总结》在MySQL中LONGBLOB是一种二进制大对象(BLOB)数据类型,用于存储大量的二进制数据,:本文主要介绍MySQL中查询和展示LO... 目录前言1. 查询 LONGBLOB 数据的大小2. 查询并展示 LONGBLOB 数据2.1 转换为十

使用SpringBoot+InfluxDB实现高效数据存储与查询

《使用SpringBoot+InfluxDB实现高效数据存储与查询》InfluxDB是一个开源的时间序列数据库,特别适合处理带有时间戳的监控数据、指标数据等,下面详细介绍如何在SpringBoot项目... 目录1、项目介绍2、 InfluxDB 介绍3、Spring Boot 配置 InfluxDB4、I

Java整合Protocol Buffers实现高效数据序列化实践

《Java整合ProtocolBuffers实现高效数据序列化实践》ProtocolBuffers是Google开发的一种语言中立、平台中立、可扩展的结构化数据序列化机制,类似于XML但更小、更快... 目录一、Protocol Buffers简介1.1 什么是Protocol Buffers1.2 Pro

Python实现数据可视化图表生成(适合新手入门)

《Python实现数据可视化图表生成(适合新手入门)》在数据科学和数据分析的新时代,高效、直观的数据可视化工具显得尤为重要,下面:本文主要介绍Python实现数据可视化图表生成的相关资料,文中通过... 目录前言为什么需要数据可视化准备工作基本图表绘制折线图柱状图散点图使用Seaborn创建高级图表箱线图热

MySQL数据脱敏的实现方法

《MySQL数据脱敏的实现方法》本文主要介绍了MySQL数据脱敏的实现方法,包括字符替换、加密等方法,通过工具类和数据库服务整合,确保敏感信息在查询结果中被掩码处理,感兴趣的可以了解一下... 目录一. 数据脱敏的方法二. 字符替换脱敏1. 创建数据脱敏工具类三. 整合到数据库操作1. 创建服务类进行数据库

MySQL中处理数据的并发一致性的实现示例

《MySQL中处理数据的并发一致性的实现示例》在MySQL中处理数据的并发一致性是确保多个用户或应用程序同时访问和修改数据库时,不会导致数据冲突、数据丢失或数据不一致,MySQL通过事务和锁机制来管理... 目录一、事务(Transactions)1. 事务控制语句二、锁(Locks)1. 锁类型2. 锁粒

Qt中实现多线程导出数据功能的四种方式小结

《Qt中实现多线程导出数据功能的四种方式小结》在以往的项目开发中,在很多地方用到了多线程,本文将记录下在Qt开发中用到的多线程技术实现方法,以导出指定范围的数字到txt文件为例,展示多线程不同的实现方... 目录前言导出文件的示例工具类QThreadQObject的moveToThread方法实现多线程QC

SpringBoot集成EasyExcel实现百万级别的数据导入导出实践指南

《SpringBoot集成EasyExcel实现百万级别的数据导入导出实践指南》本文将基于开源项目springboot-easyexcel-batch进行解析与扩展,手把手教大家如何在SpringBo... 目录项目结构概览核心依赖百万级导出实战场景核心代码效果百万级导入实战场景监听器和Service(核心

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

pandas数据的合并concat()和merge()方式

《pandas数据的合并concat()和merge()方式》Pandas中concat沿轴合并数据框(行或列),merge基于键连接(内/外/左/右),concat用于纵向或横向拼接,merge用于... 目录concat() 轴向连接合并(1) join='outer',axis=0(2)join='o