动态分配内存函数:malloc(),calloc(),realloc()总结

C 标准函数库提供了许多函数来实现对堆上内存管理,其中包括:malloc函数,free函数,calloc函数和realloc函数。使用这些函数需要包含头文件stdlib.h。它们的声明如下:

void * malloc(int n);
void free (void * p);
void *calloc(int n,int size);
void * realloc(void * p,int n);

以上函数对内存的操作都是堆上的内存,生命周期:程序员主掌生死,free()函数结束。

1.malloc函数

malloc函数可以从堆上获得指定字节的内存空间,其函数声明如下:

void * malloc(int n);

其中,形参n为要求分配的字节数。如果函数执行成功,malloc返回获得n个字节的内存空间的首地址;如果函数执行失败,那么返回值为NULL。

由于 malloc函数值的类型为void型指针,因此,需要将其值类型强行转换后赋给任意类型指针,这样就可以通过操作该类型指针来操作从堆上获得的内存空间。

需要注意的是,malloc函数分配得到的内存空间是未初始化的。(有可能申请到的内存空间之前被用过,里面存在垃圾数据,不初始化程序有安全隐患)

因此,一般在使用该内存空间时,要调用另一个函数memset来将其初始化为全0。memset函数的声明如下:

void * memset (void * p,int c,int n) ;
 该函数可以将指定的内存空间按字节单位置为指定的字符c。其中,p为要清零的内存空间的首地址,c为要设定的值,n为被操作的内存空间的字节长度。如果要用memset清0,变量c实参要为0。

malloc函数和memset函数的操作语句一般如下:

int * p=NULL;
p=(int *)malloc(sizeof(int));
if(p==NULL)
    printf(“Can’t get memory!\n”);
memset(p,0,siezeof(int));
注意:通过malloc函数得到的堆内存必须使用memset函数来初始化。c++中的new 使用更方便,1.会自动计算数据类型所需内存大小 2.返回相应类型指针值,无需type cast.

2.free()函数

从堆上获得的内存空间在程序结束以后,系统不会将其自动释放(java有垃圾回收机制,会自动释放,c/c++没有),需要程序员来自己管理。一个程序结束时,必须保证所有从堆上获得的内存空间已被安全释放,否则,会导致内存泄露。

free函数可以实现释放内存的功能。其函数声明为:

void free (void * p);

由于形参为void指针,free函数可以接受任意类型的指针实参。

但是,free函数只是释放指针的指向的内容,表明指向的内存可以被别人使用,而该指针仍然指向原来指向的地方,此时,指针为野指针,如果此时操作该指针会导致不可预期的错误。安全做法 是:在使用free函数释放指针指向的空间之后,将指针的值置为NULL。

语句如下:

free(p);
p=NULL;

3.calloc函数

calloc函数的功能与malloc函数的功能相似,都是从堆分配内存。其函数声明如下:

void *calloc(int n,int size);

函数返回值为void型指针。如果执行成功,函数从堆上获得size X n的字节空间,并返回该空间的首地址。如果执行失败,函数返回NULL。

该函数与malloc函数的一个显著不同时是,calloc函数得到的内存空间是 经过初始化的,其内容全为0。

int * p=NULL;
p=(int *)malloc(sizeof(int));
memset(p,0,siezeof(int));
与这段效果一样

int *p=NULL;
p = (int *)calloc(1,siezeof(int));
calloc函数适合为数组申请空间(会自动初始化),可以将size设置为数组元素的空间长度,将n设置为数组的容量。
提示:calloc函数的分配的内存也需要自行释放。

4. realloc函数

realloc函数的功能比malloc函数和calloc函数的功能更为丰富,可以实现内存分配和内存释放的功能,其函数声明如下:
void * realloc(void * p,int n);
其中,指针p必须为指向堆内存空间的指针,即由malloc函数、calloc函数或realloc函数分配空间的指针。realloc函数将指针 p指向的内存块的大小改变为n字节。如果n小于或等于p之前指向的空间大小,那么。保持原有状态不变。

如果n大于原来p之前指向的空间大小:

1.如果 当前连续内存块足够 realloc 的话,只是将p所指向的空间扩大,并返回p的指针地址。

2.如果 当前连续内存块不够长度,再找一个足够长的地方,分配一块新的内存q,并将 p指向的内容 copy到 q,返回 q。并将p所指向的内存空间删除。

这样也就是说 realloc 有时候会产生一个新的内存地址 有的时候不会。所以在分配完成后。我们需要判断下 p 是否等于 q。并做相应的处理。

这里有点要注意的是要避免 p = (int *)realloc(p,2048); 这种写法。有可能会造成 realloc 分配失败后,p原先所指向的内存地址丢失。

p = (int *)malloc(sizeof(int)*24);
memset(p,0,sizeof(int)*24);
q = (int *)realloc(p,sizeof(int)*48); //将p指向内存空间从24int扩充到48个int

if(q != NULL)//realloc函数执行成功,将p释放,q为新内存指针
{
  free(p);
  p = NULL;
}

注意:使用malloc函数,calloc函数和realloc函数分配的内存空间都要使用free函数或指针参数为NULL的realloc函数来释放。

下面要注意的几点是:
函数malloc()和calloc()都可以用来动态分配内存空间。 malloc()函数有一个参数,即分配的内存空间的大小,malloc在分配内存的时候会保留一定的空间用来记录分配情况,分配的次数越多,这些记录占 用的空间就越多。另外,根据malloc实现策略的不同,malloc每次在分配的时候,可能分配的空间比实际要求的多些,多次分配会导致更多的这种浪 费,当然,这些都跟malloc的实现有关; calloc()函数有两个参数,分别为元素的个数和每个元素的大小,这两个参数的乘积就是要分配的内存空间的大小。如果调用成功,它们都将返回所分配内 存空间的首地址。
函数malloc()和calloc()的主要区别是前者不能初始化所分配的内存空间,而后者可以。
realloc()可以对给定的指针所指的空间进行扩大或者缩小,无论是扩张或者缩小,原有内存中的内容将保持不变。当然,对于缩小,则被缩小的那一部分的内容会丢失。
realloc()并不保证调整后的内存空间和原来的内存空间保持同一内存地址,相反,relloc返回的指针很可能指向一个新地址。所以在代码中,我们必 须将relloc的返回值,重新赋值给p : p=(int *) realloc (p,sizeof(int)*15);

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值