结构体内存对齐

 

//结构体大小,存在内存对齐问题

 

例如:

①:我们先定义一个结构体A,并将其初始地址设为100

 结构体内存对齐

我们普通认为,其内存申请应该这样申请,a先申请1个字节大小空间,b再申请4个字节大小空间,结构体A 5个字节。

如图:

 结构体内存对齐

而实际情况却非如此:

 结构体内存对齐    

那我们来看看结构体真正的内存申请情况:

 结构体内存对齐


因为计算机读取数据时,一般标准化44个字节读取,所以为了节约时间,我们浪费3个字符大小的内存地址,以空间换取时间。

所以我们可以找出规律计算结构体的大小,以①为例:

 结构体内存对齐

:

 结构体内存对齐

结构体大小:和①规则相同,②的结构体大小=1+1+2+4=8个字节

结构体内存申请情况:

 结构体内存对齐


③:

 结构体内存对齐

结构体大小:和①②规则相同,③的结构体大小=1+1+2+1+3+4 = 12个字节

 

结构体内存申请情况:

 结构体内存对齐

④:

 结构体内存对齐

结构体大小:本身按照以上规则,③的结构体大小应该=1+3+4+2 = 10个字节

但是运行后,结果不匹配,正确答案为12个字节

 

现在我们来看看看结构体真正的内存申请情况:

 结构体内存对齐

 

这是按照以上规则,画出来的内存图,看似没错,可结果却不是12个字节。

 

现在我们来考虑一下假设结构体数组的存在:

 

struct  E  arr[2];

这时arr[1]的地址可能会出现这种情况:

 结构体内存对齐

 

这时候计算机以标准44个读取时,会在第三个绿色框框处出现问题,读取失败,不是我们需要的值。

所以我们为了避免这个情况,我们会给arr[1]结尾出补上两个字符,就不会出现读取错误了。

 

结构体内存申请情况,如图:

 结构体内存对齐


这时我们回头看④:我们会发现结构体大小 =1+3+4+2+2 = 12个字节

 

我们可以找出规律计算结构体的大小时避免出现以上问题:

就是在计算出结构体大小时,用所得值对结构体内最大的成员大小取余,看是否==0

如果!=0,我们给补上一个合适大小的内存,让取余==0

 

不过在实际工作操作时,一般会直接在结构体上补上一个空白成员并注释。

如:

 结构体内存对齐


最后我们总结一下结构体内存对齐的规则:

①:每一行和前面行总和比,不能被整除,则补上。

②:算出总共大小后和单个最大比,不能被整除,则补上。

 

我们结束前用两个结构体来验证我们的对齐规则。

①:

 结构体内存对齐

用对齐规则来计算:①:4+8=12 ;②:单个最大为int a或者int c都为4个字节,12%4==0

结果正确

将上图展开,可得:

 结构体内存对齐


②:

 结构体内存对齐

用对齐规则来计算:①:8+4=12 ;②:单个最大为double a8个字节,

12%8!=0,所以补上4个字节,(12+4%8==0

结果为16,正确。

 

相关文章:

  • 2021-06-09
  • 2022-12-23
  • 2022-12-23
  • 2022-12-23
  • 2022-12-23
  • 2021-04-07
  • 2022-01-12
猜你喜欢
  • 2021-10-08
  • 2021-08-09
  • 2022-12-23
  • 2022-12-23
  • 2021-06-25
  • 2021-11-04
相关资源
相似解决方案