malloc和new和delete异常处理浅谈
(2016-07-11 23:33:10)
标签: new和malloc异常处理 |
1.在C语言中,使用malloc来在堆上开辟内存,这里128k是个界限,小于128k通过系统调用brk()函数来分配内存,大于128k通过mmap()映射到虚拟空间,后期详解malloc内存,这里简单提一下。
如果malloc申请失败,在C语言中会返回一个NULL,
int *p = (int*)malloc(sizeof(int));
if(p == NULL)
{
}
如上程序这样,malloc申请失败会返回一个NULL指针。
2.在C++中,使用new来在堆上开辟内存,底层其实调用的是malloc,C++中的new操作符在分配内存失败时默认的操作是抛出一个内置的异常,而并不是直接返回空指针;这样的话,再把返回值与空指针比较,就没有什么意义了;因为,C++抛出异常之后,就直接跳出new操作符所在的那一行代码,而不再执行后续的代码行了,所以,对new操作符返回值的判断代码就执行不到了;
int *p =new int;
if(p == NULL)
{
}
C++中,这样写也就没有什么意义了。
当然,标准C++也提供了抑制抛出异常的方法,如下这里我总结了三个方法
a.
nothrow
如下代码
int *p = new(std::nothrow) int; if(NULL == p) { }
如果申请失败会返回一个NULL;
b.
try...catch异常处理机制
try { int *p = new int; } catch (...) { }可以这样来写,当new失败要能够扑捉异常
c.
typedef void (*new_handler)();
new_handler set_new_handler(new_handler p) throw();当new申请申请失败的时候会调用用户指定 的函数,
void Bad_alloc()
{
cout<<"mem fail"<<endl;
std::abort();
}
int main()
{
set_new_handler(Bad_alloc);
int *p = new int[1024*1024*1024*1024*1024*1024*1024*1024*1024*1024*1024*1024*1024];
return 0;
}
严格意义来说,如果operator new无法分配足够空间时,我们希望不断调用new-handler函数,直到找到足够内存为止。
需要重写operator new()函数
3.下面来看一下new[],new,delete,delete[]
通过new申请空间,需要delete来释放内存
int *p = new int;
delete p;
如果申请一个数组
int *p = new int[10];
delete []p;
这里需要注意,对于我们的内置类型,当我们申请数组的时候,需要记录申请数组的个数,C++面向对象的语言,申请空间的时候不仅需要开辟内存,还需要调用对象的构造函数,delete的时候,不仅需要需要释放内存,还需要调用对象的虚构函数。因此对于我们的自定类型申请数组的时候,会在开辟空间的前四个字节记录当前对象的个数。只有这样delete的时候才知道需要调用多少次析构函数。
注:new,delete和new[],[]delete需要成套的使用。