realloc 的空间比原始空间小时,保存的数据会发生改变么?

realloc一个比malloc分配还小新的空间时,新的空间中数值还是原来的值么?下面先分配了100个int空间,初始化并realloc后缩小到20,那20个int空间保... realloc 一个比malloc 分配还小新的空间时,新的空间中数值还是原来的值么?

下面先分配了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 就有这个特性么?新分配的空间比原来的小时,指针就还是原始指针么?
展开
 我来答
XIAOLKT
2015-05-26 · TA获得超过6.5万个赞
知道大有可为答主
回答量:5503
采纳率:64%
帮助的人:1869万
展开全部
  • 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 广告
实时数仓处理是我们北京万山数据科技有限公司数据处理能力的核心之一。它基于先进的流处理技术,能够实时捕获、处理和分析海量数据,确保数据的时效性和准确性。通过构建高效的实时数据管道,我们能够实现数据的即时入库与查询,为业务决策提供强有力的支持。... 点击进入详情页
本回答由万山数据提供
bhtzu
2014-03-18 · TA获得超过1.1万个赞
知道大有可为答主
回答量:8088
采纳率:85%
帮助的人:4229万
展开全部
realloc函数,如果原来的内存后面还有足够多剩余内存,或者,重新分配小于原来的大小,realloc还是返回原来内存的地址;
因此地址不会变化,但后面的区域成为自由区域,也就是常说的“野指针”。野指针地址在没有被重新使用和占用的情况下,值是不变的,你按照野指针强制访问,会得到原有的值。
但野指针地址内的值无法保障,如果你在realloc之后malloc一堆指针,那么无法保障这个野指针的值不变,可能会被重新占用。
你的情况,不止前20个没有变化,你输出*(p1+21)应该也是原来的值。
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
松甜恬0Je4ba
推荐于2016-07-18 · TA获得超过2.6万个赞
知道大有可为答主
回答量:7475
采纳率:100%
帮助的人:3447万
展开全部
realloc函数说明中有啊:

指针名=(数据类型*)realloc(要改变内存大小的指针名,新的大小)。
新的大小一定要大于原来的大小,不然的话会导致数据丢失!

先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。

也就是 如果比原来小 会丢失数据,至于指针是否发生变化,小的时候不会发生变化的。要变大的话,则有可能
本回答被提问者和网友采纳
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
磨辑偶4667
2015-05-29 · TA获得超过2162个赞
知道小有建树答主
回答量:465
采纳率:0%
帮助的人:452万
展开全部
新分配的空间比原来的小时,指针就不是原始指针

如果重新分配成功则返回指向被分配内存的指针,否则返回空指针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.而原来的内存块保持不变。
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
晰美酒窝
2015-05-19 · TA获得超过6732个赞
知道大有可为答主
回答量:4544
采纳率:27%
帮助的人:1556万
展开全部
这个realloc函数说明中有的

指针名=(数据类型*)realloc(要改变内存大小的指针名,新的大小)。新的大小一定要大于原来的大小,不然的话会导致数据丢失!

先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将mem_address返回,

如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝到新分配的内存区域,而后释放原来mem_address所指内存区域

(注意:原来指针是自动释放,不需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。

也就是 如果比原来小 会丢失数据,至于指针是否发生变化,小的时候不会发生变化的。要变大的话,则有可能
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
收起 更多回答(6)
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式