在空间看到别人的疑问引起了我的兴趣,刚好是我感兴趣的话题,就写一下。为了别人的疑问,也发表在qq空间里。因为下班比较晚,10点才到家,发表的也晚。其实是个简单的问题。 

直接用实例和内存图说明:

#include <iostream>

using std::cout;
using std::cin; struct stu
{
char sex;
int length;
char name[];
}; void main()
{
stu stu0;
stu0.sex = 'f';
stu0.length = ;
stu0.name[] = 'a';
stu0.name[] = 'b';
stu0.name[] = 'c';
stu0.name[] = 'd';
stu0.name[] = 'e';
stu0.name[] = 'f';
stu0.name[] = 'g';
stu0.name[] = 'h';
stu0.name[] = 'i';
stu0.name[] = 'j'; int size = sizeof(stu0);
stu *p_stu = &stu0;
cout << size;
cin.get();
}

结果为:20字节。 显然是有内存对齐的过程。

在vs下设置断点,切换反汇编,之后通过指针看到struct的内存位置,再去查看vs提供的内存查看器:
与程序内容相对应,可以清晰看到内存怎么分布的。
前四字节存储char型-66,之后四字节存储int型01然后10字节是char数组,后面补齐2字节,共20字节。显然,这里内存补齐的单位是 - 4
 
上面是以4为单位补齐,因为最大的是int型,四字节。

再看下面的例子:

#include <iostream>

using std::cout;
using std::cin; struct stu
{
char sex;
double length;
char name[];
}; void main()
{
stu stu0;
stu0.sex = 'f';
stu0.length = 1.1;
stu0.name[] = 'a';
stu0.name[] = 'b';
stu0.name[] = 'c';
stu0.name[] = 'd';
stu0.name[] = 'e';
stu0.name[] = 'f';
stu0.name[] = 'g';
stu0.name[] = 'h';
stu0.name[] = 'i';
stu0.name[] = 'j'; int size = sizeof(stu0);
stu *p_stu = &stu0;
cout << size;
cin.get();
}

结果为32。
说明是以8(double)为单位补齐 ,第一个char补为8位,第二个double8位,之后char数组10位,后面补6位。共32位。

所以,总结下,struct内存补齐机制,就是以最大的数据类型为单位来补齐。数组是连续存储在一段空间的。
知道上面我总结的一点,结合vs内存图,就会算,struct内存大小了。

你会发现,我文章一直纠结以什么单位来内存补齐。
因为这个单位是最关键的问题。
找到了内存补齐的单位,就自然会计算了。

我也就是通过小实验验证了单位为,struct中最大数据类型这一观点。

至于为什么要作内存补齐,当然是编译器为了方便cpu寻址而做的优化。如果不内存补齐,xxxxxxx(中间省略几千字)寻址会有多麻烦你知道吗?

C struct结构体内存对齐问题的更多相关文章

  1. C++ struct结构体内存对齐

    •小试牛刀 我们自定义两个结构体 A 和 B: struct A { char c1; char c2; int i; double d; }; struct B { char c1; int i; ...

  2. 【APUE】Chapter17 Advanced IPC & sign extension & 结构体内存对齐

    17.1 Introduction 这一章主要讲了UNIX Domain Sockets这样的进程间通讯方式,并列举了具体的几个例子. 17.2 UNIX Domain Sockets 这是一种特殊s ...

  3. 关于结构体内存对齐方式的总结(#pragma pack()和alignas())

    最近闲来无事,翻阅msdn,在预编译指令中,翻阅到#pragma pack这个预处理指令,这个预处理指令为结构体内存对齐指令,偶然发现还有另外的内存对齐指令aligns(C++11),__declsp ...

  4. struct结构体内存大小

    一. 基本原则 1. struct中成员变量的声明顺序,与成员变量对应的内存顺序是一致的: 2. struct本身的起始存储地址必须是成员变量中最长的数据类型的整倍数,注意是最长的数据类型,而不是最长 ...

  5. [C/C++] 结构体内存对齐用法

    一.为什么要内存对齐 经过内存对齐之后,CPU的内存访问速度大大提升; 内存空间按照byte划分,从理论上讲似乎对任何类型的变量的访问可以从任何地址开始,但实际情况是在访问特定变量的时候经常在特定的内 ...

  6. [C/C++] 结构体内存对齐:alignas alignof pack

    简述: alignas(x):指定结构体内某个成员的对齐字节数,指定的对齐字节数不能小于它原本的字节数,且为2^n; #pragma pack(x):指定结构体的对齐方式,只能缩小结构体的对齐数,且为 ...

  7. C语言-结构体内存对齐

    C语言结构体对齐也是老生常谈的话题了.基本上是面试题的必考题.内容虽然很基础,但一不小心就会弄错.写出一个struct,然后sizeof,你会不会经常对结果感到奇怪?sizeof的结果往往都比你声明的 ...

  8. c 结构体内存对齐详解

    0x00简介 首先要知道结构体的对齐规制 1.第一个成员在结构体变量偏移量为0的地址处 2.其他成员变量对齐到某个数字的整数倍的地址处 对齐数=编辑器默认的一个对齐数与该成员大小的较小值 vs中默认的 ...

  9. go语言结构体内存对齐

    cpu要想从内存读取数据,需要通过地址总线,把地址传输给内存,内存准备好数据,输出到数据总线,交给cpu,如果地址总线只有8根,那这个地址就只有8位可以表示[0,255]256个地址,因为表示不了更多 ...

随机推荐

  1. 类Arrays

    package p2; import java.util.Arrays; import java.util.List; public class ArraysDemo { public static ...

  2. C# Form.Close 的释放问题

    今天使用From窗口Close后,发现From的资源还存在,并没有释放资源,只有在程序关闭的时候才去释放. Form1:button按钮 private void button1_Click(obje ...

  3. the fifth class

      1.实际比背景长,怎么做到的? 2个父级一个做头背景一个做尾背景 2.2层,每次自带背景上下是覆盖关系,如何做到 2层?,子浮动 3.标签 4.border可覆盖:margin-bottom 为负 ...

  4. 《UML大战需求分析》阅读笔记5

    流程分析利器三,顺序图. 顺序图描述的是一件事发生的顺序,按照时间的发展,事情的走向,其中分为角色,消息等,每个角色下面都有一条生命线,从上到下,从左到右,依次进行事件,没有事情的时候用虚线表示,而有 ...

  5. 在Linux上安装最新版java的JDK

    之前写过一篇关于MC建服的文章(http://www.cnblogs.com/apollospotatolikett/p/6149042.html),文章中使用的JDK不是最新的版本,当时没有细说如何 ...

  6. jsp实现回车登录

    <body onkeydown="if(event.keyCode==13){login()}"> 内容0...... </body> 注:body里面加上 ...

  7. git push --help

    git-push(1) Manual Page NAME git-push - Update remote refs along with associated objects SYNOPSIS gi ...

  8. 最常用的reset代码

    /*css reset code */ /**** 文字大小初始化,使1em=10px *****/body { font-size:12px;} /*字体边框等初始化*/body,div,dl,dt ...

  9. [转]MySQL关键性能监控(QPS/TPS)

    原文链接:http://www.cnblogs.com/chenty/p/5191777.html 工作中尝尝会遇到各种数据库性能调优,除了查看某条SQL执行时间长短外,还需要对系统的整体处理能力有更 ...

  10. svn 版本库的创建和配置

    1.创建SVN版本库 mkdir trunk svnadmin create /root/trunk/svntest #这里是路径和即将创建的版本库名称 2.配置svn cd /root/trunk/ ...