C语言结构体对齐

1. 结构体对齐

要点

  1. 变量只能存储在他的长度的整数倍地址上
  2. 结构体整体对齐跟他的最长的字段整数倍对齐

栗子1

struct Example1 {
 char a;  //1个字节
 int c;   //4个字节
 short b; //2个字节
};

std::cout << sizeof(Example1 ) << std::endl;   // 12
std::cout << alignof(Example1) << std::endl;   // 4

int只能存储的内存序号:0,4,8,12,...
short只能存储的内存序号:0,2,4,6,...

内存序号存储大小
0char-a
1null
2null
3null
4int-c
5int-c
6int-c
7int-c
8short-b
9short-b

现在a、c、b三个变量总共占了10个字节,但是遵循第二点结构体整体对齐跟他的最长的字段的整数倍对齐,所以结构体总体应该是int=4的倍数,也就是4,8,12...
因此,最后这个结构体的大小是12个字节。对代码进行调试,监视内存窗口可以看到:
在这里插入图片描述

2. pack

要点

  1. 变量只能存储在他的min(长度,pack)的整数倍地址上
  2. 结构体整体对齐跟他的min(最长的字段,pack)整数倍对齐

栗子2

#pragma pack(2)
struct Example2 {
  char a;  //1个字节
  int c;   //4个字节
  short b; //2个字节
};

std::cout << sizeof(Example2) << std::endl;   // 8  
std::cout << alignof(Example2) << std::endl;  // 2

int原来只能存储的内存序号:0,4,8,12,...->因为min(4, 2) = 2, 现在变成0,2,4,6,...
short还是只能存储的内存序号:0,2,4,6,...

内存序号存储大小
0char-a
1null
2int-c
3int-c
4int-c
5int-c
6short-b
7short-b

现在a、c、b三个变量总共占了8个字节,但是遵循第二点结构体整体对齐跟他的min(最长的字段,pack)整数倍对齐,所以结构体总体应该是pack = 2的倍数,也就是2,4,6,8,...
因此,最后这个结构体的大小是8个字节。

对阿秀网站栗子上的解释

C++八股基础语法02

要点

  1. 变量只能存储在他的长度的整数倍地址上
  2. 结构体整体对齐跟他的最长的字段整数倍对齐
// alignas 生效的情况

struct Info {
  uint8_t a;  //1个字节
  uint16_t b; //2个字节
  uint8_t c;  //1个字节
};

std::cout << sizeof(Info) << std::endl;   // 6个字节  
std::cout << alignof(Info) << std::endl;  // 2

uint8_t只能存储的内存序号:0,1,2,3,...
uint16_t只能存储的内存序号:0,2,4,6,...

内存序号存储大小
0uint8_t -a
1null
2uint16_t-b
3uint16_t-b
4uint8_t-c

null代表内存填充为空

现在a、b、c三个变量总共占了5个字节,但是遵循第二点结构体整体对齐跟他的最长的字段整数倍对齐,所以结构体总体应该是uint16_t大小的倍数,也就是2,4,6,8,...
因此,最后这个结构体的大小是6个字节。

3. alignas与alignof

c++11以后引入两个关键字 alignas 与 alignof 。其中alignof可以计算出类型的对齐方式,alignas可以指定结构体的对齐方式。

要点

  1. 变量只能存储在他的长度的整数倍地址上【这一点和普通版没有任何区别】
  2. 结构体整体对齐跟他的max(最长的字段,alignas指定长度)整数倍对齐

因为alignas只能指定比默认值,也就是结构体最长字段,更大的值。所以对齐大小要么是默认值,要么是比默认值大的值

struct alignas(4) Info2 {
  uint8_t a;  //1
  uint16_t b; //2
  uint8_t c;  //1
};

std::cout << sizeof(Info2) << std::endl;   // 8  4 + 4
std::cout << alignof(Info2) << std::endl;  // 4

uint8_t只能存储的内存序号:0,1,2,3,...
uint16_t只能存储的内存序号:0,2,4,6,...

内存序号存储大小
0uint8_t -a
1null
2uint16_t-b
3uint16_t-b
4uint8_t-c

现在a、b、c三个变量总共占了5个字节,但是遵循第二点结构体整体对齐跟他的max(最长的字段,alignas指定长度)整数倍对齐,所以结构体总体应该是alignas = 4大小的倍数,也就是4,8,...
因此,最后这个结构体的大小是8个字节。

后面几个调试的例子运行结果如图:
在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值