C++对齐杂注__declspec(align(#))和#pragma pack的区别

2023-10-17 15:38

本文主要是介绍C++对齐杂注__declspec(align(#))和#pragma pack的区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在研究一个算法时, 我们多采用时空复杂度来衡量优劣, 然而实际生产环境中高效的代码往往不仅取决于优秀的算法,考虑硬件执行、缓存命中等特点对代码进行深入优化也很重要, 这其中就包括了内存对齐----"用空间换时间", 指令的对齐优化编译器已经帮我们考虑了, 而数据变量的对齐就需要程序员去设计了.

一般来讲, 访问未对齐的内存(misaligned access)轻则影响效率, 重则引发异常(包括指令执行异常和结构体对齐不正常引发的逻辑BUG等), 总之内存对齐是程序员尤其是C/C++程序员不得不重视的一个问题. MSVC++提供了两种对齐杂注,__declspec(align(#))和#pragma pack, 而在C++11标准中也加入了描述符alignas用于对齐控制, 本文主要简单解释下前面两种杂注的区别, 权当作者学习笔记.

这里我们要声明一个概念, 那就是对齐大小, 一般来说简单变量(内建类型)的对齐大小就是本身的大小, 其他变量(用户定义的复合变量, 如struct)则取决于声明具体类型时的具体杂注.

简单来讲, 以作用于结构体struct为例, #pragma pack用于控制结构体成员间的对齐, 而__declspec(align(#))则用于控制整个struct结构的对齐.

#pragma pack(n)要求n为1、2、4、8或16, 在未来出现256位变量的时候估计就可以指定32了.

1.第一个成员永远放置在[0...Size1)的大小为Size1的内存块(即[0]号内存开始的位置), 当只有一个成员或者所有成员的对齐大小都一致的时候不需要对齐, 整个结构的大小就是那些成员的大小总和, 此时n指定多少均不会改变struct整体占用大小.

2.否则就预留[0...Size1...n)的大小为n内存块(其中[Size1...n)这块内存是补齐出来的), 注意第一个成员是否完全占用这块大小为n的内存还要看之后的成员.

3.然后如果前面预留的[Size1...n)补齐内存块中间存在一处的内存地址是第二个成员对齐大小的倍数, 那么第二个成员就放置在这个内存地址开始的位置否则就放在[n]号内存开始的位置.

4.以此类推,重复上述步骤, 注意最后一个成员占用的大小要么为整个结构体成员中最大的对齐大小(当此值小于n时), 要么为n(当n小于整个结构体成员中最大的对齐大小时).

__declspec(align(#))则只要求#为2的整数次幂, 作用主要有两个方面

1.在结构体成员按#pragma pack确定内存布局之后, 在末尾填充内存使得整个结构的大小sizeof(struct)至少是#的整数倍.

2.作用于变量声明时, 强制要求编译器将变量放置在地址是#的整数倍的内存位置上, 这点在调用Native API等要求严格对齐的方法时十分重要, 当然动态分配时需要使用诸如_aligned_malloc等分配器才能确保正确对齐.

最后多说一句, 毋庸置疑windows编程最好最详细的文档库就是MSDN, 没有之一(推荐优先英文版, 一是有些文档没有中文版本, 二是由于机器翻译有些地方中文反而难以理解), 上面也做了MSDN相关文档的链接, 以方便有需要的读者查阅.

这篇关于C++对齐杂注__declspec(align(#))和#pragma pack的区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

Python中@classmethod和@staticmethod的区别

《Python中@classmethod和@staticmethod的区别》本文主要介绍了Python中@classmethod和@staticmethod的区别,文中通过示例代码介绍的非常详细,对大... 目录1.@classmethod2.@staticmethod3.例子1.@classmethod

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

Golan中 new() 、 make() 和简短声明符的区别和使用

《Golan中new()、make()和简短声明符的区别和使用》Go语言中的new()、make()和简短声明符的区别和使用,new()用于分配内存并返回指针,make()用于初始化切片、映射... 详细介绍golang的new() 、 make() 和简短声明符的区别和使用。文章目录 `new()`

Python中json文件和jsonl文件的区别小结

《Python中json文件和jsonl文件的区别小结》本文主要介绍了JSON和JSONL两种文件格式的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下... 众所周知,jsON 文件是使用php JSON(JavaScripythonpt Object No

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

结构体和联合体的区别及说明

《结构体和联合体的区别及说明》文章主要介绍了C语言中的结构体和联合体,结构体是一种自定义的复合数据类型,可以包含多个成员,每个成员可以是不同的数据类型,联合体是一种特殊的数据结构,可以在内存中共享同一... 目录结构体和联合体的区别1. 结构体(Struct)2. 联合体(Union)3. 联合体与结构体的

什么是 Ubuntu LTS?Ubuntu LTS和普通版本区别对比

《什么是UbuntuLTS?UbuntuLTS和普通版本区别对比》UbuntuLTS是Ubuntu操作系统的一个特殊版本,旨在提供更长时间的支持和稳定性,与常规的Ubuntu版本相比,LTS版... 如果你正打算安装 Ubuntu 系统,可能会被「LTS 版本」和「普通版本」给搞得一头雾水吧?尤其是对于刚入

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

python中json.dumps和json.dump区别

《python中json.dumps和json.dump区别》json.dumps将Python对象序列化为JSON字符串,json.dump直接将Python对象序列化写入文件,本文就来介绍一下两个... 目录1、json.dumps和json.dump的区别2、使用 json.dumps() 然后写入文