1. 结构体中最后一个数组长度为零
typedef struct _ex_mng
{
unsigned int type;
unsigned int oper;
char data[0];
}ex_mng_t;
最近在项目常用到这样子的一个结构体,最后一个成员为可变长的数组。
char data[0]中的data并不是指针,是一个偏移量,这个偏移量指向的是a、b后面紧接着的空间,代表了该结构体后面数据的起始地址,所以它其实并不占用任何空间,因此sizeof(struct _ex_mng) = 8。
使用起来非常方便,创建时,malloc一段结构体大小加上可变长数据长度的空间给它,可变长部分可按数组的方式访问,释放时,直接把整个结构体free掉就可以了。
例子:
ex_mng_t *ex_mng;
int datalen = 100;
/* 申请内存(结构体以及data的内存)*/
ex_mng = (ex_mng_t *)malloc(sizeof(ex_mng_t) + datalen );
ex_mng->type = 1;
ex_mng->oper = 1;
strcpy(ex_mng->data, "success!");
在结构体中,我们定义了0长度的数组,按理对ex_mng->data进行赋值时是属于越界访问,但是我们把结构体后面的datalen个长度的空间也一起申请了,所以该访问是合法的!
2. 结构体中最后一个成员为指针
typedef struct _ex_mng
{
unsigned int type;
unsigned int oper;
char *data;
}ex_mng_t;
而如果是指针的话,使用时候需要初始化,并且占用空间,sizeof(struct _ex_mng) = 12(32位)/sizeof(struct _ex_mng) = 12(64位)。
使用方法一: ex_mng_t *ex_mng;
int datalen = 100;
/* 申请内存(结构体) */
ex_mng = (ex_mng_t *)malloc(sizeof(ex_mng_t));
ex_mng->type = 1;
ex_mng->oper = 1;
/* 申请内存(data) */
ex_mng->data = malloc(sizeof(char)* datalen );
strcpy(ex_mng->data, "success!");
/* 释放内存(data)*/
free(ex_mng->data);
/* 释放内存(结构体)*/
free(ex_mng);
使用方法二:
ex_mng_t *ex_mng;
int datalen = 100;
/* 申请内存(结构体+data) */
ex_mng = (ex_mng_t *)malloc(sizeof(ex_mng_t) + sizeof(char)*datalen);
ex_mng->type = 1;
ex_mng->oper = 1;
/* ex_mng->data指向结构体后的内存 */
ex_mng->data = ((char *)ex_mng ) + sizeof(ex_mng_t);
strcpy(ex_mng->data, "success!");
/* 释放内存(结构体+data)*/
free(ex_mng);
注意:结构体中最后一个数组长度为零,有些编译器会报错无法编译,可以改成char data[1],或则char data[]。 总之,结构体最后使用0或1的长度数组的原因,主要是为了方便的管理内存缓冲区,当使用指针时候(使用方法一时),不能分配一段连续的的内存,会增加内存的碎片化。