c/c++ 中的内存对齐

32位linux 下:

class A

{

char c1 ;

} ;

sizeof(A) == 1 而非 4,这点有点出乎意料!按照对齐要求,在32位环境中默认的对齐是按照4字节来的,但是这里似乎没有表现出4字节的对齐要求!class B这样定义

class B

{

char c1 ;

short s1 ;

};

这个时候sizeof(B) == 4 而非 3 ,又体现出了4字节对齐的准则,对齐像是有选择性的!

其实对齐是在一定的条件下才会产生的,当类或者结构体中 data members 的类型长度不一致的时候才会产生对齐操作。这个对齐操作是编译器来自动完成的,当然按多少个字

节对齐还是程序员本身能控制的!在程序的首行使用#pragma pack(N)或者在编译的时候加上-fpack-struct=N (N代表安多少字节对齐)即可改变对齐方式。

在class A中,只有一种类型的data member ,char类型的长度都是1,并没有出现类型长度不一致的情况,所以不会出现对齐操作!但是对于class B则就不一样了,short类型的

长度和char类型的长度是不一致的,这就触发了编译器的对齐操作,所以sizeof(B) == 4而非 3,以下这个例子可以佐证这个说法。

class C

{

char c1,c2,c3,c4 ;

char c5 ;

};

sizeof(C) == 5 class C五个成员都是char类型,类型长度一致,不触发对齐操作。

既然知道了对齐操作的触发条件,那么按照4来对齐到底是什么意思呢?class B的大小sizeof(B)==4,这个因对齐而产生的填充字节被填充的位置有两中情况。第一中情况:在

c1和s1之间填充了这个字节,第二种情况:在s1之后填充这个字节,而c1和s1之间是紧挨着的!    

程序输出事实证明,填充是第一种情况下的填充!如下例:

class D

{

public:

char c1 ;

short s1 ;

};

&c1 == 0xbff23cfc
&s1 == 0xbff23cfe

从结果可以看出,0xbff23cfe - 0xbff23cfc == 1也就是说中间空了一个字节,不然的话,s1的地址应该从0xbff23cfd开始的。也就是说,对齐是按照对象或者结构体中 data

 members 中类型长度最大的一个来进行补齐操作的,注意这里的类型指的是内置类型,比如int,char等!下面一个例子是个佐证:

class E

{

char c1 ;

short s1 ;

char c2 ;

};

则sizeof(E) == 6而非8,但是默认的4字节对齐这个4字节貌似在对齐中没起到什么作用,其实默认4还是有作用的,一旦data members 中的类型长度超过了4 那么就只按照4来

进行对齐处理比如double 就是8个字节,但是程序不会按照最长的double来进行对齐操作而是按照4来进行对齐处理。下面是一个佐证:

class F

{

double d1 ;

int i1 ;

};

sizeof(F) == 12 而不是16

所以对于对齐的长度可以这么说:在类或者结构体的data members中类型长度如果有超过4(默认情况下)的,则按照4来进行对齐处理,如果没有超过4的,则按照类型长度最大

的来处理!




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
回答: 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 ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值