内存对齐 大端字节,序小端字节序验证

空结构体:对于空结构体,就是只有结构体这个模子,但里面却没有元素的结构体。

例:

typedef struct student

{

}std;

这种空结构体的模子占一个字节,sizeof(std)=1。


柔性数组:

结构体中最后一个元素可以是一个大小未知的数组,称作柔性数组成员,规定柔性数组前面至少有一个元素.

typedef struct student

{

int i;

char arr[];     //柔性数组成员

}std;

sizeof(std)=4;

sizeof求取该结构体大小是不包含柔性数组的大小,柔性数组不管有没有大小都不计入结构体的大小,可以通过动态内存为它实现内存分配。


spacer.gif内存对齐 大端字节,序小端字节序验证_第1张图片

内存对齐:

对于字(自然边界是偶数地址),双字(自然边界是能被4整除的地址),四字(自然边界是能被8整除的地址)本身就是对齐的。为什么要对齐呢?这是因为对于对齐的内存只需要一次内存访问,对于未对齐的内存,处理器要两次内存访问。

spacer.gif内存对齐 大端字节,序小端字节序验证_第2张图片

未对齐:一个字或者双字跨越了4字节边界,或者双字跨越了8字节边界,需要两次内存访问。


对于结构体,联合体在计算其大小时要考虑其内存对齐,以结构中所占字节数最大的类型类型对齐。

例:

struct test

{

    char a1;

    short b2;

    char  c3;

    int d4;

};

因为其中int类型所占字节最多,所以以4字节对齐,内存分配方式如下,总共占12个字节:

spacer.gif内存对齐 大端字节,序小端字节序验证_第3张图片

分析:因为以4字节对齐,首先char a1占一个字节存到00处,short b2 是字占两个字节 ,以偶数地址对齐,所以不能直接存到01 02 上,而应该存到02 03上,而空出来的01就会被浪费掉。同理,当char c3存到04上后,对于int i,双字必须要存到4的倍数的地址上,就只能存到08 09 0a 0b上,空出来的05 06 07会被浪费掉。


struct test

{

    char a1;

    char  c3;

    short b2;  

    int d4;

};

同样的,以4字节对齐,总共占8个字节:

这是因为a1存到00,c3存到01,b2刚好存到02 03上,以偶数地址对齐,d4也刚好从04开始存储,以4字节对齐,没有浪费内存。



大端字节序:高字节存储在低地址,低字节存储在高字节处

小端字节序:低字节存储在低地址,高字节存储在高地址处

计算机的最小存储单位是字节,一个字节占8bit位。

以int为例:

例如:1的二进制码是 

00000000 00000000 00000000 00000001

写成十六进制形式 :00 00 00 01这就代表了4个字节,而内存是从低地址到高地址的,这样就产生了两种存储方式。

spacer.gif内存对齐 大端字节,序小端字节序验证_第4张图片

因为存储方式方式的不同,读取时也就产生了两种方式。

一般读取是从低地址向高地址读取,为了将存储与读取统一取来,所以采用小端存储,这样的话,低字节存到低地址,高字节存到高地址。

验证方法:

1、可以声明一个int a=1,再声明一个char *p=&a;因为a占4个字节,char *一次向后访问一个字节,所以如果是小端的话在电脑上输出*p应该是1.

2、也可以用union,因为union是内存公用,所以声明一个int a,char c;对a赋值为1,输出c观察结果。

有些问题如果不考虑大小端的话是根本想不明白的。

你可能感兴趣的:(内存对齐,大小端验证)