柔性数组
C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做柔性数组(flexible array)成员。
typedef struct st_type
{
int i;
int a[0]; //柔性数组成员
// 有些编译器会报错无法编译可以改成:int a[];
}type_a;
特点
- 结构中的柔性数组成员前面必须至少一个其他成员。
- sizeof 返回的这种结构大小不包括柔性数组的内存。
- 包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小。
#include <stdio.h>
#include <stdlib.h>
typedef struct st_type
{
int i;
int a[0];
}type_a;
int main() {
type_a ta = { 0 };
printf("%d\n", sizeof(type_a)); //4
printf("%d\n", sizeof(ta)); //4
type_a* pta = (type_a*)malloc(sizeof(type_a) + 10 * sizeof(int));
printf("%d\n", sizeof(pta)); //8
free(pta);
pta = NULL;
return 0;
}
在malloc(sizeof(type_a) + 10 * sizeof(int))
中,后面的10 * sizeof(int)
即40个字节(10个整型大小)就是给a数组准备的内存空间。
柔性数组的好处(与另一种类似柔性数组作用的写法对比)
#include <stdio.h>
#include <stdlib.h>
typedef struct st_type
{
int i;
int* pArr;
}type_a;
int main() {
type_a* pta = (type_a*)malloc(sizeof(type_a));
pta->pArr = (int*)malloc(10 * sizeof(int));
free(pta->pArr);
pta->pArr = NULL;
free(pta);
pta = NULL;
return 0;
}
咋一看,好像与柔性数组差不多,但柔性数组却有下面这些好处:
- 方便内存释放。
如果我们的代码是在一个给别人用的函数中,你在里面做了二次内存分配,并把整个结构体返回给用户。用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free,所以你不能指望用户来发现这个事。所以,如果我们把结构体的内存以及其成员要的内存一次性分配好了,并返回给用户一个结构体指针,用户做一次free就可以把所有的内存也给释放掉(可以看到上面我写的代码演示,就先释放了里面的pArr才行)。
- 这样有利于访问速度。
连续的内存有益于提高访问速度,也有益于减少内存碎片(对于给成员动态内存分配内存,这样的话很大可能与其它成员的内存不是相连续的,这样会导致可能中间有些内存不够大,而且还可能一直用不上,会被浪费掉)。