realloc 的空间比原始空间小时,保存的数据会发生改变么?
下面先分配了100 个int 空间,初始化并realloc 后缩小到20,那20 个int 空间保存的还是开始初始化的值么?
#include "stdlib.h"
#include "stdio.h"
main(){
int *p1;
int i;
p1 = (int *)malloc(100*sizeof(int));
for(i = 0; i < 100; ++i){
p1[i] = i;
}
printf("PT:%u\n",p1);
p1 = (int *)realloc(p1, 20*sizeof(int));
for(i = 0; i < 20; ++i){
printf("%d\t",p1[i]);
}
printf("\nPT:%u\n",p1);
free(p1);
}
答案是的,而且指针的位置也没有发生改变,
realloc 就有这个特性么?新分配的空间比原来的小时,指针就还是原始指针么? 展开
realloc函数说明中有啊
指针名=(数据类型*)realloc(要改变内存大小的指针名,新的大小)。
新的大小一定要大于原来的大小,不然的话会导致数据丢失!
先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
也就是 如果比原来小 会丢失数据,至于指针是否发生变化,小的时候不会发生变化的。要变大的话,则有可能详细说明及注意要点
1、如果有足够空间用于扩大mem_address指向的内存块,则分配额外内存,并返回mem_address 这里说的是“扩大”,我们知道,realloc是从堆上分配内存的,当扩大一块内存空间时, realloc()试图直接从堆上现存的数据后面的那些字节中获得附加的字节,如果能够满足,自然天下太平。也就是说,如果原先的内存大小后面还有足够的空闲空间用来分配,加上原来的空间大小= newsize。那么就ok。得到的是一块连续的内存。
2、如果原先的内存大小后面没有足够的空闲空间用来分配,那么从堆中另外找一块newsize大小的内存。 并把原来大小内存空间中的内容复制到newsize中。返回新的mem_address指针。(数据被移动了)。 老块被放回堆上。 例如: #include <malloc.h> void main() { char *p,*q; p = (char * ) malloc (10); q=p; p = (char * )
realloc (q,20); //A ………………………… } 在这段程序中我们增加了指针q,用它记录了原来的内存地址p。这段程序可以编译通过,但在执行到A行时,如果原有内存后面没有足够空间将原有空间扩展成一个连续的新大小的话,realloc函数就会以第二种方式分配内存,此时数据发生了移动,那么所记录的原来的内存地址q所指向的内存空间实际上已经放回到堆上了!这样就会产生q指针的指针悬挂,如果再用q指针进行操作就可能发生意想不到的问题。所以在应用realloc函数是应当格外注意这种情况。 3、返回情况 返回的是一个void类型的指针,调用成功。(这就在你需要的时候进行强制类型转换) 返回NULL,当需要扩展的大小(第二个参数)为0并且第一个参数不为NULL,此时原内存变成了“freed(游离)”的了。 返回NULL,当没有足够的空间可供扩展的时候,此时,原内存空间的大小维持不变。 4、特殊情况 如果mem_address为null,则realloc()和malloc()类似。分配一个newsize的内存块,返回一个指向该内存块的指针。 如果newsize大小为0,那么释放mem_address指向的内存,并返回null。 如果没有足够可用的内存用来完成重新分配(扩大原来的内存块或者分配新的内存块),则返回null.而原来的内存块保持不变。
realloc使用总结
1. realloc失败的时候,返回NULL 2. realloc失败的时候,原来的内存不改变,不会释放也不会移动 3. 假如原来的内存后面还有足够多剩余内存的话,realloc的内存=原来的内存+剩余内存,realloc还是返回原来内存的地址; 假如原来的内存后面没有足够多剩余内存的话,realloc将申请新的内存,然后把原来的内存数据拷贝到新内存里,原来的内存将被free掉,realloc返回新内存的地址 4. 如果size为0,效果等同于free()。这里需要注意的是只对指针本身进行释放,例如对二维指针**a,对a调用realloc时只会释放一维,使用时谨防内存泄露。 5. 传递给realloc的指针必须是先前通过malloc(), calloc(), 或realloc()分配的 6.传递给realloc的指针可以为空,等同于malloc。
free:
原型: void free(void *ptr) 功 能: 释放已分配的块
2024-11-14 广告
因此地址不会变化,但后面的区域成为自由区域,也就是常说的“野指针”。野指针地址在没有被重新使用和占用的情况下,值是不变的,你按照野指针强制访问,会得到原有的值。
但野指针地址内的值无法保障,如果你在realloc之后malloc一堆指针,那么无法保障这个野指针的值不变,可能会被重新占用。
你的情况,不止前20个没有变化,你输出*(p1+21)应该也是原来的值。
指针名=(数据类型*)realloc(要改变内存大小的指针名,新的大小)。
新的大小一定要大于原来的大小,不然的话会导致数据丢失!
先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
也就是 如果比原来小 会丢失数据,至于指针是否发生变化,小的时候不会发生变化的。要变大的话,则有可能
如果重新分配成功则返回指向被分配内存的指针,否则返回空指针NULL。
当内存不再使用时,应使用free()函数将内存块释放。
注意:这里原始内存中的数据还是保持不变的。
原型:externvoid*realloc(void*mem_address,unsignedintnewsize);
用法:#include<stdlib.h>有些编译器需要#include<malloc.h>
功能:改变mem_address所指内存区域的大小为newsize长度。
说明:如果重新分配成功则返回指向被分配内存的指针,否则返回空指针NULL。
当内存不再使用时,应使用free()函数将内存块释放。
注意:这里原始内存中的数据还是保持不变的。
realloc的语法是:指针名=(数据类型*)realloc(newsize),(数据类型*)表示指针.
特性及注意要点:
1、如果有足够空间用于扩大mem_address指向的内存块,则分配额外内存,并返回mem_address
这里说的是“扩大”,我们知道,realloc是从堆上分配内存的,当扩大一块内存空间时,realloc()试图直接从堆上现存的数据后面的那些字节中获得附加的字节,如果能够满足,自然天下太平。也就是说,如果原先的内存大小后面还有足够的空闲空间用来分配,加上原来的空间大小=newsize。那么就ok。得到的是一块连续的内存。
2、如果原先的内存大小后面没有足够的空闲空间用来分配,那么从堆中另外找一块newsize大小的内存。
并把原来大小内存空间中的内容复制到newsize中。返回新的mem_address指针。(数据被移动了)。
3、返回情况
返回的是一个void类型的指针,调用成功。(这就再你需要的时候进行强制类型转换)
返回NULL,当需要扩展的大小(第二个参数)为0并且第一个参数不为NULL,此时原内存变成了“freed(游离)”的了。
返回NULL,当没有足够的空间可供扩展的时候,此时,原内存空间的大小维持不变。
4、特殊情况
如果mem_address为null,则realloc()和malloc()类似。分配一个newsize的内存块,返回一个指向该内存块的指针。
如果newsize大小为0,那么释放mem_address指向的内存,并返回null。
如果没有足够可用的内存用来完成重新分配(扩大原来的内存块或者分配新的内存块),则返回null.而原来的内存块保持不变。
指针名=(数据类型*)realloc(要改变内存大小的指针名,新的大小)。新的大小一定要大于原来的大小,不然的话会导致数据丢失!
先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,
如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域
(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
也就是 如果比原来小 会丢失数据,至于指针是否发生变化,小的时候不会发生变化的。要变大的话,则有可能