跟我一起学C++之从C到C++(结构体内存对齐)

1.什么是内存对齐

(1)      编译器为每个“数据单元”按排在某个合适的位置上。

(2)      C、C++语言非常灵活,它允许你干涉“内存对齐”。也就是可以人为的设置编译器的对齐方式。

2.为什么要对齐

性能原因:在对齐的地址上访问数据快。如果是字节对齐方式存储的话,CPU读取的时候只需要进行一个总线周期即可全部读取完毕,如果不对齐的话,对于32位的系统,CPU读取的时候一般架构都是从偶地址开启的,一次只能读取4个字节,因此需要两个总线周期才能完成,并且还需要进行寄存器里面的数据合并,还需要进行数据移位那么这样的效率就是很低的。

3.如何对齐

(1)      第一个数据成员放在offset为0的位置

(2)      其它成员对齐至min(sizeof(member),#pragma pack所指定的值)的整数倍。

(3)整个结构体也要对齐,结构体总大小对齐至各个成员中最大对齐数的整数倍。

 

代码示例:

//main.cpp
#include <iostream>
using namespace std;
#include <stdio.h>

#pragma pack(2)//修改对齐数
struct Test
{
	char a;//1个字节
	double b;//8个字节
	char c;//1个字节
};
#pragma pack()

//对齐规则
//1.第一个成员与结构体变量的偏移量为0。也就是&test = &test.a;
//2.其它成员要对齐到某个数字(对齐数)的整倍数的地址
//3.对齐数取编译器预设的一个对齐整数与该成员大小的较小值,VS中是默认值是8,所以上述打印出来就是24,取值可以是1,2,4,8,16
//4.结构体总大小为最大对齐数的整数倍


int main(void)
{
	Test test;
	//&test = &test.a;
	char *p= (char*)&test;
	//cout<<p<<endl;
	printf("结构体起始地址:%p\n", p);
	p = &test.a;
	printf("a起始地址:%p\n", p);
	p = (char *)&test.b;
	printf("b起始地址:%p\n", p);
	p = &test.c;
	printf("c起始地址:%p\n", p);

	cout<<sizeof(Test)<<endl;
	return 0;
}

运行结果:



以上例结构体为例,取值范围在VS中和在linux系统中的g++编译器,不同的对齐数的起止偏移地址,以及占用字节数如下所示:

编译器

VS

G++

对齐数

起始地址--结束地址

占用字节数

起始地址--结束地址

占用字节数

1

a:0--0

b:1—9

c:10-10

10

a:0--0

b:1—9

c:10-10

10

2

a:0--1

b:2—10

c:11-12

12

a:0--1

b:2—10

c:11-12

12

4

a:0--3

b:4—11

c:12-15

16

a:0--3

b:4—11

c:12-15

16

8

a:0--7

b:8—15

c:16-24

24

不支持

不支持

16

a:0--15

b:16—31

c:32-47

48

不支持

不支持

 


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
回答: C语言中的结构体在内存布局时会进行内存对齐内存对齐是为了提高访问效率和处理器的性能。不同的编译器和平台有不同的默认对齐规则,但通常情况下,结构体的成员会按照其大小和类型进行对齐,即每个成员的地址都是对齐的。 在引用中的程序中,使用了sizeof运算符来计算结构体A、B和C的大小。可以看到,结构体的大小是按照成员的大小和对齐规则来计算的。结构体A中只有一个char类型成员和一个int类型成员,所以大小是5字节。结构体B中有一个char类型成员、一个int类型成员和一个double类型成员,所以大小是16字节。结构体C中有一个char类型成员、一个int类型成员、一个double类型成员和一个char类型成员,所以大小是24字节。 在引用中的例子中,结构体stu1嵌套了结构体stu2,这种嵌套的情况也会影响内存对齐。具体的对齐规则可以根据编译器和平台的不同而有所差异。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [C++ 结构体内存对齐](https://blog.csdn.net/cpp_learner/article/details/119246994)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] - *2* *3* [【C/C++内存对齐(超详细,看这一篇就够了)](https://blog.csdn.net/weixin_48896613/article/details/127371045)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT0_1"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值