在看php源码时,分析变量大小时遇到了问题,结构体所占用的大小并不是简单的将里边成员所占大小相加,而是有字节对齐的问题,先看一个小例子:
#include <stdio.h>
int main(){
struct _s{
char a;//1字节
int b;//4字节
long c;//8字节
void* d;//8字节
int e;//4字节
char* f;//8字节
} s;
printf("struct type size:%d\n",sizeof(s));
return 1;
}
这个会输出33吗,经过试验,输出为40。下面就讲一下字节对齐。
结构体是一种数据类型,里面存放着不同数据类型的成员,不同的成员所占的内存空间是不同的,那系统是怎么给结构体变量分配内存呢,通过上面例子可以得出不是顺序存放的。其实是按字节对齐存放的,也就是按结构体成员中占内存最多的字节数为标准,其他成员在分配内存时都要与这个长度对其,下面画图表示一下上面的例子。
64位机器上long类型为8字节,指针也为8字节,所以该结构体中占内存最大的为8字节,char a 与int b之间会空出3个字节,int e后边也会空出4个字节,所以总共为40个字节。
总结一下,所有的成员在分配内存时都要与所有成员中占内存最多的数据类型所占内存空间的字节数对其,比如这个字节数为N,理论上所有成员在分配内存时都是紧接在前一个变量后面以此填充的,如果一行中剩下的空间不足以填充某成员变量,则该成员变量在分配内存时会另起一行分配。
分配内存时也和变量的放置的位置有关系,比如上边的例子,如果是以下这样的
#include <stdio.h>
int main(){
struct _s{
char a;//1字节
long c;//8字节
int b;//4字节
void* d;//8字节
int e;//4字节
char* f;//8字节
}s;
printf("struct type size:%d\n",sizeof(s));
return 1;
}
int b放在了long c后面,运行出来的结果为48,发生了变化,用图解释一下
所以在声明结构体类型时,各类型成员的前后位置会对结构体类型定义的变量大小产生影响,没有规律的定义会增加系统给结构体分配的字节数,降低了内存分配的效率,但这种影响几乎是可以忽略不计的,所以在写程序时尽量成员类型所占字节数从大到小或者从小到大写。