论calloc和malloc的区别

来源:互联网 发布:荷塘月色淘宝论坛首页 编辑:程序博客网 时间:2024/05/01 13:45
函数malloc()calloc()都可以用来动态分配内存空间,但两者稍有区别。
malloc()函数有一个参数,即要分配的内存空间的大小:
void*malloc(size_tsize);
calloc()函数有两个参数,分别为元素的数目和每个元素的大小,这两个参数的乘积就是要分配的内存空间的大小。
void*calloc(size_tnumElements,size_tsizeOfElement);
如果调用成功,函数malloc()和函数calloc()都将返回所分配的内存空间的首地址。
函数malloc()和函数calloc() 的主要区别是前者不能初始化所分配的内存空间,而后者能。如果由malloc()函数分配的内存空间原来没有被使用过,则其中的每一位可能都是0;反之, 如果这部分内存曾经被分配过,则其中可能遗留有各种各样的数据。也就是说,使用malloc()函数的程序开始时(内存空间还没有被重新分配)能正常进,但经过一段时间(内存空间还已经被重新分配)可能会出现问题。
函数calloc() 会将所分配的内存空间中的每一位都初始化为零,也就是说,如果你是为字符类型或整数类型的元素分配内存,那麽这些元素将保证会被初始化为0;如果你是为指针类型的元素分配内存,那麽这些元素通常会被初始化为空指针;如果你为实型数据分配内存,则这些元素会被初始化为浮点型的零。
malloc用于申请一段新的地址,参数size为需要内存空间的长度,:
char* p;
p=(char*)malloc(20);
callocmalloc相似,参数nelem为申请地址的单位元素长度,elsize为元素个数,:
char* p;
p=(char*)calloc(sizeof(char),20);
这个例子与上一个效果相同
realloc是给一个已经分配了地址的指针重新分配空间,参数ptr为原有的空间地址,newsize是重新申请的地址长度
:
char* p;
p=(char*)malloc(sizeof(char)*20);
p=(char*)realloc(p,sizeof(char)*40);
注意,这里的空间长度都是以字节为单位。 
C语言的标准内存分配函数:malloccallocreallocfree等。
malloccalloc的区别为1块与n块的区别:
malloc调用形式为(类型*)malloc(size):在内存的动态存储区中分配一块长度为“size”字节的连续区域,返回该区域的首地址。
calloc调用形式为(类型*)calloc(nsize):在内存的动态存储区中分配n块长度为“size”字节的连续区域,返回首地址。
realloc调用形式为(类型*)realloc(*ptrsize):将ptr内存大小增大到size
free的调用形式为free(void*ptr):释放ptr所指向的一块内存空间。
C++中为new/delete函数。
malloc的功能是确保内存空间使用字节。比如限定100,函数使用方法:
  (char *)malloc(100)   
超过100就会出错。
程序例:
#include <stdlib.h>
#include <stdio.h>
//#include <malloc.h>
void main( void )
{
 char *str;
 str = (char *)malloc(100);
 if( str == NULL )
  printf( "不能确保内存领域。/n" );
 else
  printf( "可以确保100字节内存。/n" );
 
 free( str ); 
 printf( "解放内存。/n" );
 getchar();
}
------------------------------------------------------------------
calloc则限定了每个要素的内存使用字节量
例:(char *)calloc(4,100);
程序例:
#include <stdlib.h>
#include <stdio.h>
//#include <malloc.h>
void main( void )
{
 char *str;
 str = (char *)calloc(4,100);
 if( str == NULL )
  printf( "不能确保内存领域。/n" );
 else
  printf( "可以确保4x100字节内存。/n" );
 free( str );
 printf( "解放内存。/n" );
 getchar();
}
首先看个问题程序(这里用的是TC编译器)
#include "stdlib.h"
#include "stdio.h"
void main()
{
   int *i;
   i=(int *)malloc(sizeof(int));
   *i=1;
   *(i+1)=2;
   printf("%x|%d/n",i,*i);
   printf("%x|%d",i+1,*(i+1));
}
输出的结果是:
8fc|1
8fe|2
这个程序编译通过,运行正常,说它有问题,问题出在哪呢?
首先通过malloc,建了一个大小为2的堆,
i指向的地址是8fci1指向的地址是8fc+sizeof(int)=8fe
但是地址8fe是不受保护的,因为它不是机器分配给i1的,随时会被其他变量占用。
正确的做法是
#include "stdlib.h"
#include "stdio.h"
void main()
{
   int *i;
   i=(int *)malloc(sizeof(int));
   *i=1;
   i=(int *)realloc(i,2*sizeof(int));
   *(i+1)=2;
   printf("%x|%d/n",i,*i);
   printf("%x|%d",i+1,*(i+1));
}
realloc 可以对给定的指针所指的空间进行扩大或者缩小,无论是扩张或是缩小,原有内存的中内容将保持不变。当然,对于缩小,则被缩小的那一部分的内容会丢失。realloc 并不保证调整后的内存空间和原来的内存空间保持同一内存地址。相反,realloc 返回的指针很可能指向一个新的地址。
所以,在代码中,我们必须将realloc返回的值,重新赋值给 p :
p = (int *) realloc (p, sizeof(int) *15);
甚至,你可以传一个空指针(0)给 realloc ,则此时realloc 作用完全相当于malloc
int* p = (int *) realloc (0,sizeof(int) * 10);  //分配一个全新的内存空间,
这一行,作用完全等同于:
int* p = (int *) malloc(sizeof(int) * 10);
『附注:TC编译器里sizeof(int)=2VC里面sizeof(int)=4
char型在两个编译器里是一样的,都是1个字节(8位)』
callocmalloc相似,参数nelem为申请地址的单位元素长度,elsize为元素个数,:
char* p;
p=(char*)calloc(sizeof(char),20);
这个例子与上一个效果相同

原文来自:http://blog.csdn.net/jlccl/archive/2007/05/23/1622834.aspx
原创粉丝点击