realloc编辑
realloc原型是extern void *realloc(void *mem_address, unsigned int newsize);
目录
1函数简介编辑
语法
新的大小一定要大于原来的大小,不然的话会导致数据丢失!
不考虑数据内容,新的大小可大可小
头文件
#include <
stdlib.h> 有些编译器需要#include <malloc.h>,在TC2.0中可以使用alloc.h头文件
功能
先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
返回值
如果重新分配成功则返回指向被分配内存的
指针,否则返回空指针NULL。
注意
当内存不再使用时,应使用
free()函数将内存块释放。
相关函数
1
|
malloc、calloc、free、_alloca
|
2应用举例编辑
举例1
从这个例子可以看出
realloc函数的功能。
运行环境:ubuntu 12.04 GCC 4.6.3
运行结果:
malloc 0x904f008
realloc 0x904f008
0 1 2 3 4 5 6 7 8 9
realloc 0x904f008
0 1 2 3 4 5 6 7 8 9
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
#include<stdio.h>
#include<stdlib.h>
intmain()
{
inti;
int
*pn=(
int
*)
malloc
(5*
sizeof
(
int
));
printf
(
"malloc%p\n"
,pn);
for
(i=0;i<5;i++)
pn[i]=i;
pn=(
int
*)
realloc
(pn,10*
sizeof
(
int
));
printf
(
"realloc%p\n"
,pn);
for
(i=5;i<10;i++)
pn[i]=i;
for
(i=0;i<10;i++)
printf
(
"%3d"
,pn[i]);
free
(pn);
return0;
}
|
举例2
:(在TC2.0中运行通过)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
//realloc.c
#include<syslib.h>
#include<alloc.h>
main()
{
char
*p;
clrscr();
//clearscreen
p=(
char
*)
malloc
(100);
if
(p)
printf
(
"MemoryAllocatedat:%x"
,p);
else
printf
(
"NotEnoughMemory!\n"
);
getchar
();
p=(
char
*)
realloc
(p,256);
if
(p)
printf
(
"MemoryReallocatedat:%x"
,p);
else
printf
(
"NotEnoughMemory!\n"
);
free
(p);
getchar
();
return0;
}
|
3内存分配编辑
-
如果有足够空间用于扩大mem_address指向的内存块,则分配额外内存,并返回mem_address。这里说的是“扩大”,我们知道,realloc是从堆上分配内存的,当扩大一块内存空间时, realloc()试图直接从堆上现存的数据后面的那些字节中获得附加的字节,如果能够满足,自然天下太平。也就是说,如果原先的内存大小后面还有足够的空闲空间用来分配,加上原来的空间大小= newsize。那么就ok。得到的是一块连续的内存。
-
如果原先的内存大小后面没有足够的空闲空间用来分配,那么从堆中另外找一块newsize大小的内存。并把原来大小内存空间中的内容复制到newsize中。返回新的mem_address指针。(数据被移动了)。老块被放回堆上。
例如:
1
2
3
4
5
6
7
8
9
|
#include<malloc.h>
voidmain()
{
char
*p,*q;
p=(
char
*)
malloc
(10);
q=p;
p=(
char
*)
realloc
(q,20);
//A行,通过realloc扩大p的空间,并把新的地址赋值给p。
//…………………………
}
|
在这段程序中我们增加了
指针q,用它记录了原来的
内存地址p。这段程序可以编译通过,但在执行到A行时,如果原有内存后面没有足够空间将原有空间扩展成一个连续的新大小的话,
realloc函数就会以第二种方式分配内存,此时数据发生了移动,那么所记录的原来的
内存地址q所指向的内存空间实际上已经放回到堆上了!这样就会产生q
指针的
指针悬挂,即指针指向了一块没有分配给用户使用的内存,如果再用q指针进行操作就可能发生意想不到的问题。所以在应用
realloc函数是应当格外注意这种情况。
4返回情况编辑
返回NULL:当需要扩展的大小(第二个参数)为0并且第一个参数不为NULL时。此时原内存变成“
free(游离)”的了。
返回NULL:当没有足够的空间可供扩展的时候。此时,原内存空间的大小维持不变。
5特殊情况编辑
如果newsize大小为0,那么释放mem_address指向的内存,并返回NULL。
如果没有足够可用的内存用来完成重新分配(扩大原来的内存块或者分配新的内存块),则返回NULL。而原来的内存块保持不变。
6使用总结编辑
1. realloc失败的时候,返回NULL
2. realloc失败的时候,原来的内存不改变,不会释放也不会移动
3. 假如原来的内存后面还有足够多剩余内存的话,realloc的内存=原来的内存+剩余内存,realloc还是返回原来内存的地址; 假如原来的内存后面没有足够多剩余内存的话,realloc将申请新的内存,然后把原来的内存数据拷贝到新内存里,原来的内存将被
free掉,realloc返回新内存的地址