本篇文章目录
- 1. 结构体中的柔性数组
- 1.2 特点
- 1.2 柔性数组的好处(与另一种类似柔性数组作用的写法对比)
- 2. C的序列化和反序列化函数
- 2.1 sprintf 函数
- 2.2 sscanf 函数
1. 结构体中的柔性数组
C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做柔性数组(flexible array)成员。
typedef struct st_type
{int i;int a[0]; //柔性数组成员// 有些编译器会报错无法编译可以改成:int a[];
}type_a;
1.2 特点
- 结构中的柔性数组成员前面必须至少一个其他成员。
- 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)); //4printf("%d\n", sizeof(ta)); //4type_a* pta = (type_a*)malloc(sizeof(type_a) + 10 * sizeof(int));printf("%d\n", sizeof(pta)); //8free(pta);pta = NULL;return 0;
}
在malloc(sizeof(type_a) + 10 * sizeof(int))
中,后面的10 * sizeof(int)
即40个字节(10个整型大小)就是给a数组准备的内存空间。
1.2 柔性数组的好处(与另一种类似柔性数组作用的写法对比)
#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才行)。
- 这样有利于访问速度。
连续的内存有益于提高访问速度,也有益于减少内存碎片(对于给成员动态内存分配内存,这样的话很大可能与其它成员的内存不是相连续的,这样会导致可能中间有些内存不够大,而且还可能一直用不上,会被浪费掉)。
2. C的序列化和反序列化函数
说专业点就是序列化,实际上就是从文件中读取了带格式的数据,然后转换成程序中的变量或对象;反序列化则是将变量或对象转换成数据存储在文件。
2.1 sprintf 函数
函数原型:
int sprintf ( char * str, const char * format, ... );
与printf类似:
int printf ( const char * format, ... );
printf是将字符串打印到终端的屏幕上,而sprintf是将字符串数据输出到一个数组。
#include <stdio.h>
struct Stu
{char gender;int age;char name[20];
};
int main() {struct Stu s = { 'm', 20, "zhangsan" };char data[50];// 假设data数组的数据在sprintf之后的文件操作存入文件sprintf(data, "%c-%d-%s", s.gender, s.age, s.name);// ...文件操作printf("%s", data);return 0;
}
2.2 sscanf 函数
函数原型:
int sscanf ( const char * s, const char * format, ...);
实际上与scanf很类似:
int scanf ( const char * format, ... );
scanf是键盘输入数据,然后从输入缓冲区读取数据;sscanf则是从一个字符串中读取一个带格式的数据(字符串)。
#include <stdio.h>
struct Stu
{char gender;int age;char name[20];
};
int main() {char data[50];// 假设字符串数据是从文件中读取的sscanf(data, "%c-%d-%s", "m-20-zhangsan");printf("%s", data);return 0;
}
未完待续…