动态内存函数的介绍
malloc和free
1.void* malloc (size_t size)
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
-如果开辟成功,则返回一个指向开辟好的空间指针。
-如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
-返回值的类型是void*,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。
-如果参数size为0,malloc的行为是标准未定义的,取决于编译器。
2.void free(void* ptr)
free函数用来释放动态开辟的内存。
如果参数ptr指向的空间不是动态开辟的,那么free函数的行为是未定义的。
如果参数ptr是NULL指针,则函数什么事也不做。
注意:free函数只是释放指针的指向的内容,表明指向的内存可以被别人使用,而该指针仍然指向原来指向的地方,
此时,指针为野指针,如果此时操作该指针会导致不可预期的错误。安全做法 是:在使用free函数释放指针指向的空
间之后,将指针的值置为NULL。
int *p = malloc( sizeof(int));//分配一个sizeof(int)的存储空间
if (!p)//判断是否为空指针
{
printf("malloc error!\n");
return 1;
}
*p = 100;//初始化
free(p);//释放内存
p = NULL;//防止出现野指针
system("pause");
return 0;
3.void* calloc (size_t num,size_t size)
-函数的功能是为num个大小为size的元素开辟一块空间,并且把空间的每个字节初始化为0.
-与函数malloc的区别只在于calloc会在返回地址之前把申请的空间的每个字节初始化为0.
int i = 0;
int *p = calloc( 3,sizeof(int));//分配一个3*sizeof(int)的存储空间
if (!p)//判断是否为空指针
{
printf("malloc error!\n");
return 1;
}
else {
for(i=0;i<3;i++)
printf("address:%d number[%d] = %d\n", &p[i], i, p[i]);//结果为0
//使用空间
}
free(p);//释放内存
p = NULL;//防止出现野指针
system("pause");
return 0;
除了直接初始化为0之外,其他功能与malloc函数相同。
4.void* realloc (void* ptr,size_t size)
-ptr是要调整的内存大小
-size是调整之后新大小
-返回值为调整之后的内存起始位置
-这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间
-realloc在调整内存空间的是存在两种情况:
情况1:原有空间之后又足够大的空间
情况2:原有空间之后没有足够大的空间
int *p = NULL;
int *q = NULL;
p = (int * )malloc(sizeof(int) * 24);
memset(p, 0, sizeof(int) * 24);
printf("%p\n", p);
q = (int *)realloc(p, sizeof(int) * 48); //将p指向内存空间从24int扩充到48个int
if (q != NULL)//realloc函数执行成功,将p释放,q为新内存指针
{
p = q;
}
printf("%p\n", p);
free(p);
p = NULL;
system("pause");
return 0;
说明:指针p必须指向堆内存空间;
p是指指针p指向的内存块的大小为size字节。
①size小于或等于p之前指向的空间大小:保持原有状态不变。
②size大于原来p之前指向的空间大小:如果当前连续内存块足够 realloc的话,只是将p所指向的空间扩大,并返回ptr的指针地址;如果当前连续内存块不够长度,再找一个足够长的地方,分配一块新的内存q,并将 p指向的内容copy到q,返回q,并将p所指向的内存空间删除。
这样也就是说 realloc有时候会产生一个新的内存地址,有的时候不会,所以在分配完成后,我们需要判断下p是否等于 q。
优点:
相比malloc函数和calloc函数,realloc函数的功能更为丰富,可以实现动态内存的分配以及释放。共同点就是:
均需要free函数来堆内存进行释放。