new和malloc的区别
1.申请的内存所在位置
new操作符从自由存储区上为对象动态分配内存空间,而malloc从堆上动态分配内存。自由存储区是C++基于new操作符的一个抽象概念,凡是通过new进行内存申请,该内存即为自由存储区。而堆是操作系统所维护的一块特殊内存,用于程序的动态内存分配。C语言使用malloc从堆上分配内存,使用free释放对应内存。
那么自由存储区是否可以是堆(问题等价于new是否能够在堆上动态分配内存)?这取决于operator new的实现细节。事实上自由存储区不仅可以是堆,还可以是静态存储区。
2.返回类型安全性
new内存分配成功时,返回的是对象类型的指针,无需进行类型转换,因此new是符合类型安全性的操作符。而malloc内存分配成功返回void *,需要通过强转将void *转化为需要的类型。
3.内存分配失败时的返回值
new内存分配失败时,会抛出一个bac_alloc异常,不会返回NULL。malloc内存分配失败返回NULL。
4.是否需要指定内存大小
使用new申请内存分配时无需指定内存块的大小,编译器会根据类型自行计算。而malloc则需要显式地指出所需内存的大小。
5.是否调用构造/析构函数
new/delete会调用对象的构造/析构函数,而malloc则不会。
new分配对象内存的三个步骤:
(1)调用operator new()分配内存;
(2)调用构造函数来构造对象并传入初值;
(3)返回指向该对象的指针。
delete释放对象内存的两个步骤:
(1)调用析构函数;
(2)调用operator delete释放内存。
new申请失败不抛出异常的办法:
1.使用关键字nothrow
#include <new>
int main()
{
set_new_handler(newhandler);
Manager *pManager = new (std::nothrow) Manager();
if(NULL == pManager)
{
return false;
}
}
2.替换new_handler()
当new分配内存而没有足够内存时,会调用new_handler(),而缺省的new_handler()会抛出一个bad_alloc()。为了避免这个异常,可以替换一个不抛出异常的new_handler()即可。
#include <new>
void _cdecl newhandler()
{
return ;
}
int main()
{
set_new_handler(newhandler);
Manager *pManager = new (std::nothrow) Manager();
if(NULL == pManager)
{
return false;
}
}