目录
学习目标:
- 学习动态内存管理
学习内容:
一、为什么存在动态内存分配
我们知道数组的创建之后,其大小也就确定了。
如:int arr[20] = {0};
1. 空间开辟大小是固定的。
2. 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。
因此有时候我们的数组又不能被固定大小,这时候就需要动态开辟空间
二、动态内存函数的介绍
2.1 malloc
返回类型是指针类型,括号中是申请开辟的空间的大小
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
如果开辟成功,则返回一个指向开辟好空间的指针。
如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己 来决定。
如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。
2.2 free
释放掉开辟的申请的空间
要注意的几个点:
参数为NULL时什么都不做
参数不能是非动态的,且free函数的行为是未定义的。
2.3 calloc
这个是malloc的升级,开辟num个,大小为size的空间,并且把空间都初识化为0
与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。
2.4 realloc
当我们申请的空间不够用时,可以用realloc来继续申请;
realloc函数的出现让动态内存管理更加灵活。
有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时 候内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小 的调整。
ptr 是要调整的内存地址
size 调整之后新大小
返回值为调整之后的内存起始位置。
这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到 新 的空间。
realloc在调整内存空间的是存在两种情况:
情况1:原有空间之后有足够大的空间
情况2:原有空间后面没有足够连续空间开辟
则函数会在找另一个足够的地方,开辟空间。先将原有的数据拷贝一份到新位置上。原来位置的销毁。
情况1 当是情况1 的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化。
情况2 当是情况2 的时候,原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找一个合适大小 的连续空间来使用。这样函数返回的是一个新的内存地址。 由于上述的两种情况,realloc函数的使用就要注意一些。
三、常见的动态内存错误
3.1对NULL指针解引用
void test() { int *p = (int *)malloc(INT_MAX/4); *p = 20;//如果p的值是NULL,就会有问题 free(p); }
3.2 对动态开辟空间的越界访问
3.3 对非动态开辟内存使用free释放
3.4 使用free释放一块动态开辟内存的一部分
3.5 对同一块动态内存多次释放
3.6忘记释放申请开辟的动态内存
这可能会导致内存泄漏,很严重的事情
五、柔性数组
C99 中,结构中的最后一个元素允许是未知大小的数组,这就叫做『柔性数组』成员。
柔性数组的特点:
结构中的柔性数组成员前面必须至少一个其他成员。
sizeof 返回的这种结构大小不包括柔性数组的内存。
包含柔性数组成员的结构用malloc ()函数进行内存的动态分配,并且分配的内存应该大于结构的大 小,以适应柔性数组的预期大小。
柔性数组的使用:
struct S { int i; int arr[]; }; int main() { struct S s; struct S* ptr = (struct S*)malloc(sizeof(struct S) + 100 * sizeof(int)); if (ptr == NULL) { return; } ptr->i = 100; int i = 0; for (i = 0; i < 100; i++) { ptr->arr[i] = i; } for (i = 0;i < 100; i++) { printf("%d ", ptr->arr[i]); } free(ptr); ptr = NULL; return 0; }
使用柔性数组的好处:
方便内存的释放
提高访问速度
学习时间:
二零二二年十月
学习产出:
- CSDN 技术博客 1 篇