结构体struts的长度
在需要计算结构体大小的时候,涉及到的一个问题就是其对齐模数
计算机系统对基本类型数据在内存中存放的位置有限制,它们会要求这些数据的首地址的值是某个数k(通常它为4或8)的倍数,这就是所谓的内存对齐,而这个k则被称为该数据类型的对齐模数(alignment modulus)。
也就是说对齐模数就是这个数据类型占用的空间大小。
关于结构体长度的计算,我查了一下,有两种理解的方式:
方式一:
当一种类型S的对齐模数与另一种类型T的对齐模数的比值是大于1(类型S强于T)的整数,我们就称类型S的对齐要求比T强(严格),而称T比S弱(宽松)。
struct A{
short a; //k为1
int b; //k为4
char c; //k为1
double d; //k为8
}a;
ANSI C规定一种结构类型的大小是它所有字段的大小以及字段之间或字段尾部的填充区大小之和。填充区就是为了使结构体字段满足内存对齐要求而额外分配给结构体的空间。那么结构体本身也有对齐要求,ANSI C标准规定结构体类型的对齐要求不能比它所有字段中要求最严格的那个宽松,可以更严格。
也就是说,结构体的模数应该是字段中最强字段类型模数的整数倍。
所以上面的代码对应的内存布局图应该是这样:

int类型强于sort类型,则int变量的首地址应该是4的倍数;double的类型强于char,double的首地址就应该是8的倍数。
这里,padding是填充区,注意,填充区就是为了使结构体字段满足内存对齐要求而额外分配给结构体的空间。只有当结构体中的成员一种类型S的对齐模数与另一种类型T的对齐模数不一致的时候,才可能产生填充区。
在上面的结构体中,a与b之间填充了两个空间,b的类型强于c,所以b到c不需要填充空间,c的前面已经占用了8个字节,而c本身还要有1个字节的空间即位置这时到了9,所以c到d还要填充7个空间,即:

打印地址:

所以:结构体a占用的内存空间大小应该是24。
但而如果把结构体A的位置改变一下:
struct B{
short a; //模数为2
char c; //模数为1
int b; //模数为4
double d; //模数为8
}b;
那么其对应的内存分布发生了改变:

打印结果:

综上,可以得出:
- 先计算变量。结构体中变量的位置,必须是对齐模数的整数倍,不是整数倍则会分配填充区进行填充。
- 再计算结构体。结构体的长度是对齐模数和填充区的和。
但此时,还要考虑到结构体自身的对齐模数(结构体也是基本数据类型)。他的模数是#pragma pack定义的模数与结构体内部最大的基本数据类型成员长度中数值较小者。结构体的长度应该是该模数的整数倍。
如下面的示例
struct B{
int b; //模数为4
char a; //模数为1
}b;
此时,结构体中的变量的总长度为5,按上面的想法,得出此结构体的长度应该是最强类型int的倍数,则为4 * 2 = 8:

是的,就是8,但是,如果加上编译选项#pragma pack预处理指令:
#pragma pack(push, 2)
struct B{
int b; //模数为4
char a; //模数为1
}b;
#pragma pack(pop)
此时成员b的对其模数应该以2为主,a的是1,小于2,则继续以1的对齐模数为主。但结构体的模数k应该是编译编译选项与结构体内部数据类型最强的成员长度中数值中的较小者,这里是2,所以a的长度就变成了6(2的倍数):

在实际开发中,通过指定/Zp编译选项或者在代码中用#pragma pack指令来更改编译器的对齐规则。比如指定/Zpn(VC7.1中n可以是1、2、4、8、16)就是告诉编译器最大对齐模数是n。
在这种情况下,所有小于等于n字节的基本数据类型的对齐规则与默认的一样,但是大于n个字节的数据类型的对齐模数被限制为n。如果n = 1,那么结构体的大小就是各个字段的大小之和,在Moses中定义结构体的地方随处可见,这样做可以减少结构体所占用的内存空间。
#pragma pack(push, 1)
struct B{
int b; //模数为4
char a; //模数为1
}b;
//此时结构体变量b的长度就为5
#pragma pack(pop)
方式二:
- 先计算变量。根据对齐模数计算结构体变量中的起始地址,必须是对齐模数的整数倍,不是整数倍会自动补齐。
- 再计算结构体。根据结构体的对齐模数计算结构体的大小。结构体的对齐模数是#pragma pack定义的模数与结构体内部最大的基本数据类型成员长度中数值较小者。结构体的长度应该是该模数的整数倍。(但这里要注意如果有预处理指令定义时,变量的模数如果大于n,则按n的对齐规则)
按照这种方式:
struct A{
short a; //模数为2
int b; //模数为4
char c; //模数为1
double d; //模数为8
}a;
1.先计算结构体中成员的地址:
a:地址为0,是其模数2的倍数,所以不再补齐。
b:如果不对齐,则地址为2。不是模数的整数倍,应补齐2个空间(其实也就是方式一中的填充区),地址变成4。
c:此时地址为8,是1的整数倍,不再补齐。
d:如果不对齐,则地址为9。所以应补齐8个空间,地址变成16
此时结构体中成员占据的空间为24
通过打印地址也可验证:

2.计算结构体模数:
此时没有定义编译器指令,所以,结构体模数应该是其成员中最大占用空间类型的模数,也就是8,其大小为24,是8的倍数,所以结构体不再对齐。
对于这两种方式,我觉得方式二可能更好理解一些,但是要注意考虑到编译指令的问题,最后得出:
1.当没有定义编译指令n时:
结构体长度= 结构体内成员对齐后占用的空间之和(两两对比);
之后还需验证结构体地址是否是成员最强类型变量模数k的的整数倍。
2.当定义了编译指令n时:
如果类型强度小于n那么还是按照成员的对齐规则。如果是大于n则需按照n的对齐规则。
同样在之后还是要验证结构体地址是否是对齐模数的整数倍。如果不是,则结构体模数同样也要进行对齐。
结构体struts的长度的更多相关文章
- C# Struct结构体里数组长度的指定
typedef struct Point{ unsigned short x; unsigned short y; }mPoint;//点坐标 typedef struct Line{ mPoint ...
- 失落的C语言结构体封装艺术
Eric S. Raymond <esr@thyrsus.com> 目录 1. 谁该阅读这篇文章 2. 我为什么写这篇文章 3.对齐要求 4.填充 5.结构体对齐及填充 6.结构体重排序 ...
- Linux C 程序 预处理,结构体(13)
C语言预处理,结构体 C语言预处理命令1.宏定义 1.无参数宏 #define 标识符 字符串 #代表本行是编译预处理命名 习惯上,宏定义大写 代替一个字符串,介绍重复书写某个字符串的工作量 有意义的 ...
- (转)失落的C语言结构体封装艺术
目录1. 谁该阅读这篇文章 2. 我为什么写这篇文章 3.对齐要求 4.填充 5.结构体对齐及填充 6.结构体重排序 7.难以处理的标量的情况 8.可读性和缓存局部性 9.其他封装的技术 10.工具 ...
- 关于C语言中结构体中的结构体成员导致的字节对齐问题
关于结构体的字节对齐是什么,就不赘述,再此附上一篇文章,介绍字节对齐:http://www.linuxsong.org/2010/09/c-byte-alignment/ 这里的结构体字节对齐的数据类 ...
- C++结构体中sizeof
说明: 结构体的sizeof值,并不是简单的将其中各元素所占字节相加,而是要考虑到存储空间的字节对齐问题.这些问题在平时编程的时候也确实不怎么用到,但在一些笔试面试题目中出是常常出现,一.解释 现代计 ...
- C学习之结构体
结构体(struct) 结构体是由基本数据类型构成的.并用一个标识符来命名的各种变量的组合,结构体中可以使用不同的数据类型. 1. 结构体说明和结构体变量定义 在Turbo C中, 结构体也是一种数据 ...
- C#中结构体定义并转换字节数组
最近的项目在做socket通信报文解析的时候,用到了结构体与字节数组的转换:由于客户端采用C++开发,服务端采用C#开发,所以双方必须保证各自定义结构体成员类型和长度一致才能保证报文解析的正确性,这一 ...
- <摘录>字节对齐与结构体大小
说明: 结 构体的sizeof值,并不是简单的将其中各元素所占字节相加,而是要考虑到存储空间的字节对齐问题.这些问题在平时编程的时候也确实不怎么用到,但在一 些笔试面试题目中出是常常出现,对sizeo ...
随机推荐
- Gerrit与Gitlab同步配置replication&其他配置
一.Gerrit与Gitlab同步配置 当配置好gerrit环境后,还需要与现有gitlab库进行同步配置,否则会影响现有开发与打包流程. 1.安装gerrit replication插件 unzip ...
- AutoIt操作Windows GUI实现文件上传
AutoIt 一个使用类似BASIC脚本语言的免费软件,用于Windows GUI(图形用户界面)中进行自动化操作.它利用模拟键盘按键,鼠标移动和窗口/控件的组合来实现自动化任务. 官方网站: htt ...
- Update Request
public function update(UpdateAppointmentRequest $request) { try { $data = array_filter($request-> ...
- reset 单个文件 回退
git将单个文件恢复到历史版本的正确方法如下: git reset commit_id 文件路径 git checkout -- 文件路径
- plsql11.06注册码
plsql11.06注册码:Product Code(产品编号):4t46t6vydkvsxekkvf3fjnpzy5wbuhphqzserial Number(序列号):601769password ...
- C# 构造post参数一种看起来直观点的方法[转]
因为本人经常爱用C#做一些爬虫类的小东西,每次构造post参数的时候,都是直接字符串拼接的方式的,有时候改起来不太方便. 场景: 需要post一个地址 参数列表 : username:管理员 pass ...
- web应用程序开发原理
企业应用计算的演变为1.主机/哑终端的集中计算模式: 2.客户机/服务器计算模式:3.浏览器 /服务器计算模式.其中,1具有部署方面的优势,但在一台计算机中进行全部的处理,应用程序难于维护,难于 ...
- Image模块
1.创建一个新的图片 Image.new(mode,size) Image.new(mode,size,color) 2.层叠图片 层叠两个图片,img1和img2,alpha是一个介于[0,1]的浮 ...
- 关于(object sender, EventArgs e)
sender是事件源 就是指发起这个事件的对象(控件)//表示触发事件的那个控件比如说你按下按钮,那么sender就是按钮 又如:textboxchange,sender就是该textbox,在事 ...
- WINDOWS下PhoneGap(Cordova)安装笔记
1.首先下载Node.js 安装nodejs很简单直接点击安装文件下一步直至成功即可,安装notejs的同时npm也会同时安装 成功后打开notejs的命令行工具 输入“node -v”," ...