您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > C/C++/C#

深入理解C++数据类型对齐

时间:2023-07-31 14:38:29  来源:微信公众号  作者:鲨鱼编程
在C++中,为了提高内存访问效率,编译器会对某些数据类型的变量进行对齐。数据对齐是指数据存储地址要求保持一定的对齐比特,通常是内存总线宽度的整数倍。合理的对齐可以优化存储器存取,提高访问性能。

图片

对齐的原因

现代CPU在访问内存时,是以一个word(字)为访问单位,一个word大小通常为4字节或8字节。如果数据存储地址不是word大小的整数倍,就需要多次内存访问才能读取完,这会降低访问效率。
举例:一个int类型占4字节,地址为0x1004,那么读取这个int需要两次访问:第一次访问地址0x1004,第二次访问地址0x1008,两次访问才能把int读完。如果int的地址是0x1008,就是4字节对齐的,那么只需要访问一次就可以读取完,效率更高。
对齐方式的选择
在选择数据类型的对齐方式时,需要考虑多个因素,包括数据类型的大小、系统架构、编译器实现等。通常情况下,对于较小的数据类型,可以选择字节对齐;对于较大的数据类型,可以选择自然对齐或最宽基本数据类型对齐。此外,在编写跨平台的程序时,需要考虑系统架构的不同,选择合适的对齐方式,以确保程序在不同系统上的运行效果一致。

 

C++中的对齐
C++编译器会自动对结构体、类和数组等进行对齐。具体来说:
  1. 结构体和类的每个成员会根据其大小和对齐要求进行对齐
  2. 数组的每个元素会对齐到元素大小的整数倍
  3. 整型提升为与机器字大小相同的类型
以32位系统为例(word大小为4字节),结构体align的定义:
struct align {
  char a; // 1字节 
  int b; // 4字节
  double c; // 8字节 
};
结构体align的大小不是每个成员大小的简单相加,而要考虑对齐,会调整每个成员的偏移,让每个成员地址都是4的整数倍:
a偏移 0 (对齐到 0)
b偏移 4 (对齐到 4的整数倍)  
c偏移 8 (对齐到 8的整数倍)

结构体总大小是12字
又如把align中的int改为char,结构体大小就变为8字节,因为加上一个char后总大小就是8的整数倍了。

 

强制对齐
C++还提供了一些对齐属性来控制数据对齐:
  • attribute((aligned(n))): 指定数据对齐到n字节
  • attribute((packed)):取消结构体中的优化对齐
示例:
struct noalign {
  char a; 
  int b;
  double c;
} __attribute__((packed)); // 取消优化对齐

struct align16 {
  char a;
  int b; 
  double c;  
} __attribute__((aligned(16))); // 16字节对齐
通过控制对齐可以优化存储器访问,但也会增加结构体的大小,需要权衡空间和时间的效率。
对齐的影响因素
数据类型的对齐方式会直接影响结构体、类等复合数据类型的内存布局,进而影响程序的性能和可移植性。常见的对齐问题包括内存浪费、程序崩溃、数据读取错误等。
内存浪费是最常见的对齐问题之一。当数据类型的对齐方式不合适时,会导致结构体等复合数据类型中出现无用的填充字节,从而浪费内存空间。例如,对于一个包含多个char类型的变量的结构体,如果使用自然对齐,那么会出现大量的填充字节,从而浪费了内存空间。
程序崩溃是另一个常见的对齐问题。当数据类型的对齐方式不正确时,会导致程序在访问内存时出现未定义的行为,例如读取到错误的数据、访问非法的内存地址等,从而导致程序崩溃。这种情况下,通常需要重新设计数据结构,以确保数据类型的对齐方式符合要求。
数据读取错误也是一种常见的对齐问题。当数据类型的对齐方式不正确时,会导致某些数据类型的读取出现错误,例如float、double等浮点数类型。这种情况下,可能需要使用特殊的类型转换方式来保证数据的正确读取。
代码示例
下面是一个简单的代码示例,展示了数据类型对齐的影响:
#include <IOStream>

using namespace std;

struct Test {
    char a;
    int b;
    char c;
};

int mAIn() {
    Test t;
    cout << "sizeof(Test) = " << sizeof(Test) << endl;
    cout << "&t.a = " << (void*)&t.a << endl;
    cout << "&t.b = " << (void*)&t.b << endl;
    cout << "&t.c = " << (void*)&t.c << endl;
    return 0;
}
在这个示例中,定义了一个包含char、int、char类型的结构体Test。通过sizeof运算符可以获取结构体的大小,通过取地址操作可以获取结构体中各个成员变量的地址。运行程序可以得到如下输出:
sizeof(Test) = 12
&t.a = 0x7ffee2c3b1c0
&t.b = 0x7ffee2c3b1c4
&t.c = 0x7ffee2c3b1c8
可以看到,结构体Test的大小为12字节,其中有两个字节的填充。这是因为在默认情况下,编译器使用自然对齐方式,使得结构体的对齐位置是4的倍数。如果将编译器选项设置为不使用填充字节,可以得到如下输出:
sizeof(Test) = 9
&t.a = 0x7ffee2c3b1c0
&t.b = 0x7ffee2c3b1c1
&t.c = 0x7ffee2c3b1c5
可以看到,此时结构体Test的大小为9字节,没有任何填充字节。这种情况下,结构体的对齐方式是字节对齐。


Tags:C++   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系,我们将及时更正、删除。
▌相关推荐
C++常见避坑指南
C++ 从入门到放弃?本文主要总结了在C++开发或review过程中常见易出错点做了归纳总结,希望借此能增进大家对C++的了解,减少编程出错,提升工作效率,也可以作为C++开发的避坑攻略。...【详细内容】
2024-04-03  Search: C++  点击:(4)  评论:(0)  加入收藏
C++ 之父反驳白宫警告:自诞生第一天起,C++ 的目标就一直是提高安全性
整理 | 郑丽媛上个月,美国白宫国家网络主任办公室(ONCD)在一份主题为《回到基础构件:通往安全软件之路》的 19 页 PDF 报告中,呼吁开发人员停止使用容易出现内存安全漏洞的编程语...【详细内容】
2024-03-25  Search: C++  点击:(4)  评论:(0)  加入收藏
八个 C++ 开源项目,帮助初学者进阶成长
通过参与或阅读开源项目的源代码,你可以获得丰富的实践机会。实际的项目代码比简单的教程更具挑战性,可以帮助你深入理解 C++ 的各种概念和技术。1.ThreadPool一个简单的 C++1...【详细内容】
2024-03-22  Search: C++  点击:(21)  评论:(0)  加入收藏
C++多线程编程:解锁性能与并发的奥秘
今天我们将深入探讨C++中的多线程编程,揭示多线程如何解锁性能潜力,提高程序的并发性能。什么是多线程?在计算机科学中,多线程是指一个进程(程序的执行实例)中的多个线程同时执行...【详细内容】
2024-02-03  Search: C++  点击:(69)  评论:(0)  加入收藏
C++代码优化攻略
今天我们将深入探讨C++性能优化的世界。在当今软件开发的浪潮中,高性能的代码是必不可少的。无论是开发桌面应用、移动应用,还是嵌入式系统,性能都是关键。1. 选择合适的数据结...【详细内容】
2024-01-26  Search: C++  点击:(113)  评论:(0)  加入收藏
C++质数检测器的设计与实现​
质数,作为数学中的一个基本概念,一直以其独特的性质吸引着众多研究者和爱好者。质数是指大于1的自然数中,除了1和它本身以外不再有其他因数的数。在实际应用中,质数检测也扮演着...【详细内容】
2024-01-15  Search: C++  点击:(111)  评论:(0)  加入收藏
指针变量在C/C++中的内存占用
在编程领域,尤其是C和C++这类底层语言中,指针是一个核心概念,它允许程序直接操作内存地址。然而,关于指针本身在内存中占用的空间大小,却常常让初学者感到困惑。本文将深入探讨这...【详细内容】
2024-01-09  Search: C++  点击:(95)  评论:(0)  加入收藏
C++的面向对象编程:深入解析与理解
当我们谈论C++时,面向对象编程(OOP)是一个无法回避的话题。那么,C++的面向对象究竟是什么?为什么它如此重要?本文将从基本概念到实际应用,为您详细解析C++中的面向对象编程。一、面...【详细内容】
2024-01-03  Search: C++  点击:(95)  评论:(0)  加入收藏
有什么好用的C/C++源代码混淆工具?
开始使用ipaguard前言iOS加固保护是直接针对ios ipa二进制文件的保护技术,可以对iOS APP中的可执行文件进行深度混淆、加密。使用任何工具都无法逆向、破解还原源文件。对APP...【详细内容】
2023-12-29  Search: C++  点击:(118)  评论:(0)  加入收藏
C++中new与malloc:内存分配机制深度解析
本文旨在深入探讨C++中new和malloc两种内存分配机制的区别。通过对比它们在内存分配、初始化、错误处理、调用构造函数/析构函数、类型转换和使用便捷性等方面的不同,我们将...【详细内容】
2023-12-27  Search: C++  点击:(127)  评论:(0)  加入收藏
▌简易百科推荐
C++常见避坑指南
C++ 从入门到放弃?本文主要总结了在C++开发或review过程中常见易出错点做了归纳总结,希望借此能增进大家对C++的了解,减少编程出错,提升工作效率,也可以作为C++开发的避坑攻略。...【详细内容】
2024-04-03  腾讯技术工程    Tags:C++   点击:(4)  评论:(0)  加入收藏
C++ 之父反驳白宫警告:自诞生第一天起,C++ 的目标就一直是提高安全性
整理 | 郑丽媛上个月,美国白宫国家网络主任办公室(ONCD)在一份主题为《回到基础构件:通往安全软件之路》的 19 页 PDF 报告中,呼吁开发人员停止使用容易出现内存安全漏洞的编程语...【详细内容】
2024-03-25    CSDN  Tags:C++   点击:(4)  评论:(0)  加入收藏
八个 C++ 开源项目,帮助初学者进阶成长
通过参与或阅读开源项目的源代码,你可以获得丰富的实践机会。实际的项目代码比简单的教程更具挑战性,可以帮助你深入理解 C++ 的各种概念和技术。1.ThreadPool一个简单的 C++1...【详细内容】
2024-03-22  AI让生活更美好  微信公众号  Tags:C++   点击:(21)  评论:(0)  加入收藏
C# 中15个值得收藏的开源项目推荐
在开源的世界里,C# 编程语言也占有一席之地。这些开源项目涵盖了多个领域,从框架、库到工具,它们为C#开发者提供了丰富的资源和工具,帮助他们更高效地开发、测试和部署应用程序...【详细内容】
2024-03-20  程序员编程日记  微信公众号  Tags:C#   点击:(30)  评论:(0)  加入收藏
C#异步编程:Task.Run vs. async-await,掌握基础与高级用法
概述:C#中的异步编程有两主要方式:Task.Run用于在后台线程执行同步操作,而async-await更适用于清晰表达异步流程。基础用法展示了它们的简单应用,高级用法则演示了它们的结合使...【详细内容】
2024-03-09  架构师老卢  今日头条  Tags:C#   点击:(23)  评论:(0)  加入收藏
C++多线程编程:解锁性能与并发的奥秘
今天我们将深入探讨C++中的多线程编程,揭示多线程如何解锁性能潜力,提高程序的并发性能。什么是多线程?在计算机科学中,多线程是指一个进程(程序的执行实例)中的多个线程同时执行...【详细内容】
2024-02-03     AI让生活更美好  Tags:C++   点击:(69)  评论:(0)  加入收藏
C++代码优化攻略
今天我们将深入探讨C++性能优化的世界。在当今软件开发的浪潮中,高性能的代码是必不可少的。无论是开发桌面应用、移动应用,还是嵌入式系统,性能都是关键。1. 选择合适的数据结...【详细内容】
2024-01-26  AI让生活更美好  微信公众号  Tags:C++   点击:(113)  评论:(0)  加入收藏
C# 线程本地存储为什么线程间值不一样
为什么用 ThreadStatic 标记的字段,只有第一个线程拿到了初始值,其他线程都是默认值,让我能不能帮他解答一下,尼玛,我也不是神仙什么都懂,既然问了,那我试着帮他解答一下,也给后面类...【详细内容】
2024-01-26  一线码农聊技术  微信公众号  Tags:C#   点击:(67)  评论:(0)  加入收藏
C++质数检测器的设计与实现​
质数,作为数学中的一个基本概念,一直以其独特的性质吸引着众多研究者和爱好者。质数是指大于1的自然数中,除了1和它本身以外不再有其他因数的数。在实际应用中,质数检测也扮演着...【详细内容】
2024-01-15  鲨鱼编程  微信公众号  Tags:C++   点击:(111)  评论:(0)  加入收藏
C# 登顶!超越Java或非空想
整理丨诺亚出品 | 51CTO技术栈(微信号:blog51cto)近日,TIOBE编程社区公布年度编程语言,此次摘得这一桂冠的是C#。这也是C#在TIOBE二十多年评选历史中首次赢得这一年度大奖。C#虽...【详细内容】
2024-01-15    51CTO  Tags:C#   点击:(114)  评论:(0)  加入收藏
站内最新
站内热门
站内头条