关于结构体内存对齐方式的总结(#pragma pack()和alignas())
最近闲来无事,翻阅msdn,在预编译指令中,翻阅到#pragma pack这个预处理指令,这个预处理指令为结构体内存对齐指令,偶然发现还有另外的内存对齐指令aligns(C++11),__declspec(align(#))(Microsoft专用),遂去探究两者之间的不同点。
1、#pragma pack
这个指令为预处理指令,所谓与处理指令执行在程序的预处理阶段,该指令对应着编译选项/Zp,可以在vs的工程属性中设置编译选项的内存对齐,也可以利用预处理指令来设置。
#pragma pack( [ show ] | [ push | pop ] [, identifier ] , n )
预处理指令的用法如上,其中必选参数n为内存对齐值,有效值为1、4、8、16,默认值为8。可选参数中,show代表右警告消息显示当前的内存对齐值,push | pop 二者选其一,代表将当前的内存对齐值入、出栈。identifier跟随push | pop一同出入栈,作为表示使用。
2、aligns、__declspec(align(#))
aligns和alignof为c++标准的类型说明和运算符,其中aligns为内存对齐类型符,alignof为返回内存对齐值得运算符。对应还有Microsoft的msvc专用的__alignof()和__declspec(align(#))也可以达到同样的的效果。
__declspec(align(#))和aligns(#)中#为对齐值,可选的对齐值为2、4、8、16、32、64
3、区别
同样是对齐操作,两种不同的对齐操作有什么区别呢。
字面上,一个是预处理指令,另一个是类型说明符。
在相应的结构体内存对齐上,也存在差别。
所谓的内存对齐指的是变量分配到要求的内存地址上,这个地址必须是对齐值的整数倍,例如int型变量,默认的对齐值为数据长度,也就是4,分配到内存地址0x0001103F,该地址模上内存对齐值,也就是4。1103F%4=3,所以将int型变量分配到0x00011042上,该地址为对齐值得整数倍。
定义一个结构体,结构体内存对齐值为内存存储变量的最大默认对齐值
struct MyStruct
{
char member;
int a; }mystruct; sizeof(mystruct) = ,alignof(MyStruct) =
通过计算可以看出,结构体的对齐值为4,内存占用为8。结构体遵循内存对齐规则,结构体内部变量也遵循对齐规则,其中char偏移量0,int偏移量为4,两变量之间填充3字节数据。
- #pragma pack(1),再次计算sizeof和alignof
sizeof(mystruct) = 5;alignof(MyStruct) = 1
可以看出,预处理指令要求的内存对齐为1设置生效了。
- __declspec(align(2)),最小对齐值,再次计算sizeof和alignof
__declspec(align(2))
struct MyStruct
{
char member;
int a;
}mystruct;
sizeof(mystruct) = 8;alignof(MyStruct) = 4
设置内存对齐失败了。
- 用c++11标准再试试,align(2)
struct alignas() MyStruct
{
char member;
int a; }mystruct;
这下vs直接报错,'MyStruct': Alignment specifier is less than actual alignment (4), and will be ignored
通过这个报错,可以看出,类型说明符形式的内存对齐要求最小的对齐值为结构体的对齐默认值,也就是内存变量的最大对齐 值。
- __declspec(align(8)),计算sizeof和alignof
__declspec(align()) struct MyStruct
{
char member;
int a;
}mystruct;
sizeof(mystruct) = ;alignof(MyStruct) = struct alignas() MyStruct
{
char member;
int a; }mystruct;
sizeof(mystruct) = 8;alignof(MyStruct) = 8
其中char偏移为0,int 的偏移为4,中间填充3字节数据
- 将aligns(8)添加到结构体内部变量,设置内部变量的对齐值
struct MyStruct
{
alignas() char member;
alignas() int a;
}mystruct; sizeof(mystruct) = ;alignof(MyStruct) =
此时可以看出,结构体对齐值为内部变量最大对齐值8,其中char变量对齐值为8,偏移量0,int对齐值为8,偏移量为8,中间 填充7字节数据,结尾填充4字节数据。可见 alignas()对于修饰的变量有效
- aligna(16)
struct alignas() MyStruct
{
char member;
int a;
}mystruct; sizeof(mystruct) = ;alignof(MyStruct) =
sizeof(mystruct) = 16;alignof(MyStruct) = 16
其中char偏移量0,int偏移量为4,中间填充3字节数据,结尾填充8字节数据
- #pragma pack(16)
sizeof(mystruct) = 8;alignof(MyStruct) = 4
可以看出通过预编译指令设置对齐失效了。
aligna()和#pragma(1)
#pragma pack(1)
struct alignas() MyStruct
{
char member;
int a;
}mystruct; sizeof(mystruct) = ;alignof(MyStruct) = 设置了预编译指令和alignas(),遵循alignas()
4、总结
我们可以通过两种方式来设置对齐,分别是#pragam pack()和alignas()/declspce(align(#))
对于#pragma pack()我们可以设置全局对齐方式,结构体和结构体内部变量都将遵循设置的对齐参数;而alignas()对修饰的单个变量是,对齐参数有效,例:alignas()修饰结构体变量,那么结构体变量遵循该对齐参数,内部变量遵循默认的对齐参数,以上对齐方式均是对结构体变量起作用,对于数据区的其他变量不管如何设置对齐方式,还会按照默认的字节来对齐。
#pragma pack()这种对齐参数设置有最大上限,最大设置为其默认对齐参数;alignas()/declspce(align(#))这种对齐方式存在最小下限,最小下限为默认对齐参数。
对于设置两种对齐方式,优先遵循alignas()这种对齐方式。
结构体的默认对齐方式为4byte,最终的结构体对齐方式与结构体内所占最大空间的项的对齐方式一致
结构体所占空间一定为对齐参数的整数倍,也就意味着结构内部可能会存在填充的字节
关于结构体内存对齐方式的总结(#pragma pack()和alignas())的更多相关文章
- 【APUE】Chapter17 Advanced IPC & sign extension & 结构体内存对齐
17.1 Introduction 这一章主要讲了UNIX Domain Sockets这样的进程间通讯方式,并列举了具体的几个例子. 17.2 UNIX Domain Sockets 这是一种特殊s ...
- C语言中结构体内存存储方式
C语言中结构体内存存储方式 结构体的默认存储方式采用以最大字节元素字节数对其方式进行对齐,例如一个结构体中定义有char.int类型元素,则结构体存储空间按照int类型占用字节,如果还有double类 ...
- C++ struct结构体内存对齐
•小试牛刀 我们自定义两个结构体 A 和 B: struct A { char c1; char c2; int i; double d; }; struct B { char c1; int i; ...
- [C/C++] 结构体内存对齐用法
一.为什么要内存对齐 经过内存对齐之后,CPU的内存访问速度大大提升; 内存空间按照byte划分,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内 ...
- [C/C++] 结构体内存对齐:alignas alignof pack
简述: alignas(x):指定结构体内某个成员的对齐字节数,指定的对齐字节数不能小于它原本的字节数,且为2^n; #pragma pack(x):指定结构体的对齐方式,只能缩小结构体的对齐数,且为 ...
- C语言-结构体内存对齐
C语言结构体对齐也是老生常谈的话题了.基本上是面试题的必考题.内容虽然很基础,但一不小心就会弄错.写出一个struct,然后sizeof,你会不会经常对结果感到奇怪?sizeof的结果往往都比你声明的 ...
- C/C++ 结构体内存对齐
内存对齐是指的是编译器在编译的时候总是会将结构体的元素的地址放在一些合适的位置使得CPU访问这些数据的效率变得更高.首先来看下面这个例子: struct A{ char a; char b; int ...
- c 结构体内存对齐详解
0x00简介 首先要知道结构体的对齐规制 1.第一个成员在结构体变量偏移量为0的地址处 2.其他成员变量对齐到某个数字的整数倍的地址处 对齐数=编辑器默认的一个对齐数与该成员大小的较小值 vs中默认的 ...
- go语言结构体内存对齐
cpu要想从内存读取数据,需要通过地址总线,把地址传输给内存,内存准备好数据,输出到数据总线,交给cpu,如果地址总线只有8根,那这个地址就只有8位可以表示[0,255]256个地址,因为表示不了更多 ...
随机推荐
- Java - 多线程Callable、Executors、Future
http://blog.csdn.net/pipisorry/article/details/44341579 Introduction Callable接口代表一段能够调用并返回结果的代码; Fut ...
- new,malloc,GlobalAlloc具体解释
WINDOWS下最好的方式是用VirtualAlloc分配内存,他不是在堆,也不是栈,而是直接在进程的地址空间中保留一快内存.尽管用起来最不方便. 可是速度快,也最灵活 new,malloc,Glob ...
- [codeforces 618 F] Double Knapsack (抽屉原理)
题目链接:http://codeforces.com/contest/618/problem/F 题目: 题目大意: 有两个大小为 N 的可重集 A, B, 每个元素都在 1 到 N 之间. 分别找出 ...
- Asp.Net中使用水晶报表(下)
Asp.Net中使用水晶报表(下) 使用PUSH模式 我们采用下面的几步使用Push模式执行水晶报表: 1. 设计一个DataSet 2. 创建一个.rpt文件同时将其指定给上一步建立的DataS ...
- theano import error (win10 python2.7)
因为项目需要,在win10-64位电脑上配置theano.但是一直有 import error的错误,找不到解决方法.作为一个python新手,实在搞不定,请大家不吝赐教!小女子不胜感激! 按照网上的 ...
- form表单提交的时候,传过去的值是键值对的形式
效果展示 第一种需求,点击input的时候,input的value发生改变 $('.group-wrapper input').click(function(){ $(this).val(0); // ...
- NodeJS学习笔记 进阶 (4)基于express+muter的文件上传(ok)
个人总结:这篇文章主要讲了multer插件的使用,类似于formidable,可以用来处理post表单中的文件上传,读完这篇文章需要10分钟. 摘选自网络 概览 图片上传是web开发中经常用到的功能, ...
- echarts如何修改数据视图dataView中的样式
原文链接:点我 做了一个现实折线图的图表,通过右上角icon可以自由切换成柱状图,表格.在表格中遇到的一点小问题,解决方案如下: 1.场景重现 这是一个显示两个折线图的图表,一切看起来都很顺利.但是点 ...
- 学习《深度学习入门:基于Python的理论与实现》高清中文版PDF+源代码
入门神经网络深度学习,推荐学习<深度学习入门:基于Python的理论与实现>,这本书不来虚的,一上来就是手把手教你一步步搭建出一个神经网络,还能把每一步的出处讲明白.理解神经网络,很容易就 ...
- python、js 时间日期模块time
python 参考链接:https://www.runoob.com/python/python-date-time.html 时间戳 >>> print(time.time())# ...