动态内存函数的介绍
malloc
在堆区开辟空间
申请一块连续可用的空间,并返回指向这块空间的指针
如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查
free
free(p);//当动态申请的空间不再使用的时候,就应该还给操作系统
p=NULL;//彻底断开
malloc和free应该成对使用
calloc
(开辟后进行了初始化)
realloc
调整动态开辟内存的大小
1.如果p指向空间之后有足够的内存空间可以追加,则直接追加,后返回p
2.如果p指向的空间之后没有足够的内存空间可以追加,则realloc函数会重新找一个新的内存区域
开辟一块满足需要的空间,并且把原来内存中的数据拷贝回来,释放旧的内存空间
最后返回新开辟的内存空间地址
3.需要用一个新的变量来接收realloc函数的返回值
realloc第一个参数给空指针时功能和malloc一样
int *p=realloc(NULL,40);//功能等价于malloc()
常见的动态内存错误
1、对NULL空指针进行解引用操作
int* p = (int*)malloc(40);
//不一定开辟成功,要对p进行相关的判断
*p = 10;//malloc开辟空间失败,对NULL指针解引用
2、对动态开辟内存的越界访问
int* p = (int*)malloc(40);//10个int
if (p == NULL)
{
return 0;
}
int i = 0;
//越界
for (i = 0; i <= 10; i++)//访问了11个元素
{
*(p + i) = i;
}
3、对非动态开辟内存使用free释放
int a = 10;
int* p = &a;//a空间是在栈区上,不是动态开辟出来的
//...
//对非动态开辟内存使用free释放
free(p);
p = NULL;
4、使用free释放动态开辟内存的一部分
int* p = (int*)malloc(40);
if (p == NULL)
{
return 0;
}
int i = 0;
for (i = 0; i < 10; i++)
{
*p++;
}
free(p);//p变化了,不是开头了
5、对同一块内存多次释放
int* p = (int*)malloc(40);
if (p == NULL)
{
return 0;
}
//使用
//释放
free(p);
//...
free(p);//重复释放
free(p);后跟P=NULL;可避免
6、对动态开辟内存忘记释放(内存泄漏)
动态开辟的空间一定要释放,并且正确释放,不能只开辟不回收
while (1)
{
malloc(1);
}
柔性数组
结构中的最后一个元素允许是未知大小的数组,叫做【柔性数组】成员
特点:
结构中的柔性数组成员前面必须至少一个其他成员
sizeof返回的这种结构大小不包括柔性数组的内存
包含柔性数组成员的结构用malloc()函数进行内存的动态分配,并且分配的内存应该大于结构的大小,以适应柔性数组的预期大小
优点:
1.方便内存释放2.有利于访问速度