国内最全IT社区平台 联系我们 | 收藏本站
华晨云阿里云优惠2
您当前位置:首页 > php开源 > php教程 > [置顶] malloc和new的区别

[置顶] malloc和new的区别

来源:程序员人生   发布时间:2014-12-23 08:39:20 阅读次数:4129次

在申请分配动态内存时,malloc和new1直都困扰着程序员,今天就作个简单的对照,找到它们的区分:

1、主要区分

1,malloc与free是C++/C语言的标准库函数,new/delete是C++的运算符。它们都可用于申请动态内存和释放内存。

2,对非内部数据类型的对象而言,光用maloc/free没法满足动态对象的要求。对象在创建的同时要自动履行构造函数,对象在灭亡之前要自动履行析构函数。由于malloc/free是库函数而不是运算符,不在编译器控制权限以内,不能够把履行构造函数和析构函数的任务强加于malloc/free。

3,因此C++语言需要1个能完成动态内存分配和初始化工作的运算符new,以1个能完成清算与释放内存工作的运算符delete。注意new/delete不是库函数。

4,C++程序常常要调用C函数,而C程序只能用malloc/free管理动态内存,new 是个操作符,和甚么"+","-","="...有1样的地位. malloc是个分配内存的函数,供你调用的.

new是保存字,不需要头文件支持.malloc需要头文件库函数支持.new 建立的是1个对象,malloc分配的是1块内存.

简单的说: malloc,free是c的函数,而new,delete是c++的运算符 。另外,new是强迫类型的,malloc不是,需要类型转换 ,固然还有很多不同: new 可以调用构造函数在声明的时候初始化,而 malloc只是分配空间,需要在其他地方初始化 ;delete不但会释放空间,在释放前会调用析构函数 ,而且malloc需要指定分配空间大小, 而new是自动计算的 。
 

注意free()释放的是指针指向的内存!注意!释放的是内存,不是指针!这点非常非常重要!指针是1个变量,只有程序结束时才被烧毁。释放了内存空间后,原来指向这块空间的指针还是存在!只不过现在指针指向的内容的垃圾,是未定义的,所以说是垃圾。因此,前面我已说过了,释放内存后把指针指向NULL,避免指针在后面不谨慎又被解援用了。

2、扼要分析

1、malloc函数

1.1 malloc的全称是memory allocation,中文叫动态内存分配。
原型:extern void *malloc(unsigned int num_bytes); 
说明:分配长度为num_bytes字节的内存块。如果分配成功则返回指向被分配内存的指针,分配失败返回空指针NULL。当内存不再使用时,应使用free()函数将内存块释放。

1.2 void *malloc(int size); 
说明:malloc 向系统申请分配指定size个字节的内存空间,返回类型是 void* 类型。void* 表示未肯定类型的指针。C,C++规定,void* 类型可以强迫转换为任何其它类型的指针。   
备注:void* 表示未肯定类型的指针,更明确的说是指申请内存空间时还不知道用户是用这段空间来存储甚么类型的数据(比如是char还是int或...)

1.3 free
void free(void *FirstByte): 该函数是将之前用malloc分配的空间还给程序或是操作系统,也就是释放了这块内存,让它重新得到自由。

1.4注意事项

1)申请了内存空间后,必须检查是不是分配成功
2)当不需要再使用申请的内存时,记得释放;释放后应当把指向这块内存的指针指向NULL,避免程序后面不谨慎使用了它。 
3)这两个函数应当是配对。如果申请后不释放就是内存泄漏;如果无故释放那就是甚么也没有做。释放只能1次,如果释放两次及两次以上会出现毛病(释放空指针例外,释放空指针其实也等于啥也没做,所以释放空指针释放多少次都没有问题)。
4)虽然malloc()函数的类型是(void *),任何类型的指针都可以转换成(void *),但是最好还是在前面进行强迫类型转换,由于这样可以躲过1些编译器的检查。

1.5  malloc()到底从哪里得到了内存空间?

答案是从堆里面取得空间。也就是说函数返回的指针是指向堆里面的1块内存。操作系统中有1个记录空闲内存地址的链表。当操作系统收到程序的申请时,就会遍历该链表,然后就寻觅第1个空间大于所申请空间的堆结点,然后就将该结点从空闲结点链表中删除,并将该结点的空间分配给程序。

2、new运算符

2.1 C++中,用new和delete动态创建和释放数组或单个对象

动态创建对象时,只需指定其数据类型,而没必要为该对象命名,new表达式返回指向该新创建对象的指针,我们可以通过指针来访问此对象
int *pi=new int;
这个new表达式在堆区中分配创建了1个整型对象,并返回此对象的地址,并用该地址初始化指针pi 。

2.2 动态创建对象的初始化

动态创建的对象可以用初始化变量的方式初始化。
int *pi=new int(100); //指针pi所指向的对象初始化为100
string *ps=new string(10,’9’);//*ps 为“9999999999”

如果不提供显示初始化,对类类型,用该类的默许构造函数初始化;而内置类型的对象则无初始化
也能够对动态创建的对象做值初始化:
int *pi=new int( );//初始化为0
int *pi=new int;//pi 指向1个没有初始化的int
string *ps=new string( );//初始化为空字符串 (对提供了默许构造函数的类类型,没有必要对其对象进行值初始化)

2.3 撤消动态创建的对象

delete表达式释放指针指向的地址空间。
delete pi ;// 释放单个对象
delete [ ]pi;//释放数组
如果指针指向的不是new分配的内存地址,则使用delete是不合法的。

2.4 在delete以后,重设指针的值

delete p; //履行完该语句后,p变成了不肯定的指针,在很多机器上,虽然p值没有明肯定义,但依然寄存了它之前所指对象的地址,然后p所指向的内存已被释放了,所以p不再有效。此时,该指针变成了悬垂指针(悬垂指针指向曾寄存对象的内存,但该对象已不存在了)。悬垂指针常常致使程序毛病,而且很难检测出来。
1旦删除指针所指的对象,立行将指针置为0,这样就非常清楚的指明指针不再指向任何对象。(零值指针:int *ip=0;)

2.5 辨别零值指针和NULL指针

零值指针,是值是0的指针,可以是任何1种指针类型,可以是通用变体类型void*也能够是char*,int*等等。
空指针,其实空指针只是1种编程概念,就如1个容器可能有空和非空两种基本状态,而在非空时可能里面存储了1个数值是0,因此空指针是人为认为的指针不提供任何地址讯息。

2.6 new分配失败时,返回甚么?

1993年前,c++1直要求在内存分配失败时operator   new要返回0,现在则是要求operator   new抛出std::bad_alloc异常。很多c++程序是在编译器开始支持新规范前写的。c++标准委员会不想放弃那些已有的遵守返回0规范的代码,所以他们提供了另外情势的operator   new(和operator   new[])以继续提供返回0功能。这些情势被称为“无抛出”,由于他们没用过1个throw,而是在使用new的入口点采取了no throw对象
class   widget   {   ...   };

widget   *pw1   =   new   widget;//   分配失败抛出std::bad_alloc  

if   (pw1   ==   0)   ... //   这个检查1定失败

widget   *pw2   =   new   (nothrow)   widget;   //   若分配失败返回0

if   (pw2   ==   0)   ... //   这个检查可能会成功

3. malloc和new的区分

3.1 new 返回指定类型的指针,并且可以自动计算所需要大小。
比如:   
1) int *p;   
p = new int; //返回类型为int* 类型(整数型指针),分配大小为 sizeof(int);   
或:   
int* parr;   
parr = new int [100]; //返回类型为 int* 类型(整数型指针),分配大小为 sizeof(int) * 100;   
2) 而 malloc 则必须要由我们计算字节数,并且在返回后强行转换为实际类型的指针。   
int* p;   
p = (int *) malloc (sizeof(int)*128);//分配128个(可根据实际需要替换该数值)整型存储单元,并将这128个连续的整型存储单元的首地址存储到指针变量p中  
double *pd=(double *) malloc (sizeof(double)*12);//分配12个double型存储单元,并将首地址存储到指针变量pd中

3.2 malloc 只管分配内存,其实不能对所得的内存进行初始化所以得到的1片新内存中,其值将是随机的
除分配及最后释放的方法不1样之外,通过malloc或new得到指针,在其它操作上保持1致。

4.有了malloc/free为何还要new/delete?

1) malloc与free是C++/C语言的标准库函数,new/delete是C++的运算符。它们都可用于申请动态内存和释放内存。
2) 对非内部数据类型的对象而言,光用maloc/free没法满足动态对象的要求。对象在创建的同时要自动履行构造函数,对象在灭亡之前要自动履行析构函数。由于malloc/free是库函数而不是运算符,不在编译器控制权限以内,不能够把履行构造函数和析构函数的任务强加于malloc/free。
因此C++语言需要1个能完成动态内存分配和初始化工作的运算符new,和1个能完成清算与释放内存工作的运算符delete。注意new/delete不是库函数。
我们不要企图用malloc/free来完成动态对象的内存管理,应当用new/delete。由于内部数据类型的“对象”没有构造与析构的进程,对它们而言malloc/free和new/delete是等价的。
3) 既然new/delete的功能完全覆盖了malloc/free,为何C++不把malloc/free淘汰出局呢?这是由于C++程序常常要调用C函数,而C程序只能用malloc/free管理动态内存。
如果用free释放“new创建的动态对象”,那末该对象因没法履行析构函数而可能致使程序出错。如果用delete释放“malloc申请的动态内存”,结果也会致使程序出错,但是该程序的可读性很差。所以new/delete必须配对使用,malloc/free也1样。



生活不易,码农辛苦
如果您觉得本网站对您的学习有所帮助,可以手机扫描二维码进行捐赠
程序员人生
------分隔线----------------------------
分享到:
------分隔线----------------------------
关闭
程序员人生