realloc(void *__ptr, size_t __size):更改已经配置的内存空间,即更改由malloc()函数分配的内存空间的大小。
如果将分配的内存减少,realloc仅仅是改变索引的信息。
如果是将分配的内存扩大,则有以下情况:
1)如果当前内存段后面有需要的内存空间,则直接扩展这段内存空间,realloc()将返回原指针。
2)如果当前内存段后面的空闲字节不够,那么就使用堆中的第一个能够满足这一要求的内存块,将目前的数据复制到新的位置,并将原来的数据块释放掉,返回新的内存块位置。
3)如果申请失败,将返回NULL,此时,原来的指针仍然有效。
注意:如果调用成功,不管当前内存段后面的空闲空间是否满足要求,都会释放掉原来的指针,重新返回一个指针,虽然返回的指针有可能和原来的指针一样,即不能再次释放掉原来的指针。
看一下示例代码
#include<stdio.h>
#include<stdlib.h>
#include<iostream>
using namespace std;
int main()
{
int input;
int n;
int *numbers1;
int *numbers2;
numbers1 = NULL;
numbers2 = (int*)malloc(5*sizeof(int));
if(numbers2 ==NULL)
{
printf("malloc memory unsuccessful~");
exit(1);
}
printf("numbers2 addr: %8x\n ",(int)numbers2); //将指针地址转化为int
for(n=0; n<5; n++)
{
*(numbers2+n)=n;
}
for(n=0;n<5;n++)
{
cout<<*(numbers2+n);
}
printf("Enter the new size:\n");
scanf("%d",&input);
//重新分配内存空间,如果分配成功的话,就释放numbers2指针, 然后给numbers1赋值的时候,不会影响到numbers2以前的赋值。也就是记者numbers2后面的开始赋值!
//但是并没有将numbers2指针赋为NULL,也就是说释放掉的是系统分配的堆空间,
//和该指针没有直接的关系,现在仍然可以用numbers2来访问这部分堆空间,但是
//现在的堆空间已经不属于该进程的了。
numbers1=(int *)realloc(numbers2,(input+5)*sizeof(int)); //核心就是重新建立一个指针对象,然后将原来的指针成员realloc到这个对象。
if (numbers1 == NULL)
{
printf("Error (re)allocating memory");
exit(1);
}
printf("numbers1 addr: %8x\n",(int)numbers1);
for (n = 0; n<input;n++)
{
*(numbers1+5+n)=n+5;
}
for(n=0;n<5+input;n++)
{
cout<<*(numbers1+n);
}
free(numbers1);
numbers1 = NULL;
return 0;
}