c++中的set_new_handler和new_handler

转载 2016年05月31日 19:58:01

详细可参考 《effective c++》第三版 条款49:了解new_handler的行为


当operator new申请一个内存失败的时候,它会进行如下的处理步骤:
    1、如果存在客户指定的处理函数,则调用处理函数(new_handler),如果不存在则抛出一个异常。

    2、继续申请内存分配请求。
    3、判断申请内存是否成功,如果成功则返回内存指针,如果失败转向处理步骤1


为了自定义这个“用以处理内存不足”的函数new_handler,用户可以调用set_new_handler进行设置

这两个函数声明如下:

namespace std{

     typedef void (*new_handler)();

     new_handler set_new_handler(new_handler p) throw();

}

其中,new_handler是个typedef,定义一个函数指针,该函数没有参数,也没有返回值;

set_new_handler用于设置处理函数,设置p为当前处理函数,并返回之前的new_handler


当operator new无法满足内存分配需求时,它会不断调用new_handler函数,直到找到足够的内存。

因此,应该妥善设计new_handler函数,一个设计良好的new_handler必须做以下事情:

1、删除其它无用的内存,使系统具有可以更多的内存可以使用,为下一步的内存申请作准备。

实现此策略的办法是:程序一开始执行就分配一大块内存,当new_handler被调用时,将它们释放还给程序使用。

2、设置另外一个new_handler。

如果当前的new_handler不能够做到更多的内存申请操作,或者它知道另外一个new_handler可以做到,

则可以调用set_new_handler函数设置另外一个new_handler,这样在operator new下一次调用的时候,

可以使用这个新的new_handler。

3、卸载new_handler,使operator new在下一次调用的时候,因为new_handler为空抛出内存申请异常。

4、new_handler抛出自定义的异常

5、不再返回,调用abort或者exit退出程序


c++并不支持专属某一类的new_handler,但是如果需要,可以重载operator new,自己实现这个行为。

只需为class提供自己的set_new_handler和operator new即可。

在operator new中做如下事情:

1、首先调用标准的set_new_handler,自定义专属类的处理函数

2、调用global operator new,执行实际的内存分配。如果内存分配失败,刚才被安装的new_handler将被调用。

3、无论new成功还是失败,都必须在类自定义的operator new结束前恢复全局new_handler

这一部分的详细示例参考《effective c++》 条款49


转自:http://blog.csdn.net/liuxialong/article/details/6540605

———————————————————————————————————————————————————

在过去或者传统的模式中,当operator new不能满足一个内存分配请求的时候,它会返回一个空指针,但是现在如果operator new不能满足一个内存分配请求的时候,会抛出一个异常,这个时候我们可不可以自己定制处理策略呢?这正是new_handler的作用所在,具体情况在后面详细道来。

当operator new申请一个内存失败时,它会进行如下的处理步骤:
1、如果存在客户指定的处理函数,则调用处理函数(new_handler),如果不存在则抛出一个异常。new_handler的模型为:void (*new_handler)()。
2、继续申请内存分配请求。
3、判断申请内存是否成功,如果成功则返回内存指针,如果失败转向处理步骤1。

可以用如下的代码表示:
全局部分: 
void (*class_new_oom_handler)();
......

operator new执行方法中:

复制代码
void (*my_new_oom_handler)();
void *result;
    
for( ; ; )
{
    my_new_oom_handler = class_new_oom_handler;
    if( my_new_oom_handler == NULL )
        THROW_BAD_NEW; //如果没有设置自定义处理函数,则默认的处理方式为抛出一个异常

    (*my_new_oom_handler)(); //调用自定义内存异常处理函数
    result = malloc( n ); //继续申请内存
    if( result )
        return( result ); //申请成功,返回指针
}
复制代码

说了这么多,大家一定会问,那我怎么设置这个处理函数(new_handler),通过“void set_new_handler( void(*new_handler)()) throw();”进行设置,它定义在<new>标准函数库中:

namespace std
{
    void (*new_handler)();
    void set_new_handler( new_handler )throw();
}
复制代码
//error-handling function
void MemErrorHandling()
{
    std::cerr << "Failed to allocate memory.\n";
    std::abort();
}
... ...
std::set_new_handler(MemErrorHandling);
复制代码

 

现在我们知道了new操作失败后,系统地大概处理流程,以及怎么设置用户自定义处理函数,但是我们究竟可以在new_handler中做些什么处理呢?

1、删除其它无用的内存,使系统具有可以更多的内存可以使用,为下一步的内存申请作准备。
2、设置另外一个new_handler。如果当前的new_handler不能够做到更多的内存申请操作,或者它知道另外一个new_handler可以做到,则可以调用set_new_handler函数设置另外一个new_handler,这样在operator new下一次调用的时候,可以使用这个新的new_handler。
3、卸载new_handler(通过set_new_handler(0)),使operator new在下一次调用的时候,因为new_handler为空抛出内存申请异常。
4、抛出自定义异常。
5、不再返回,调用abort或者exit退出程序。

参考: 
1、http://www.bc-cn.net/Article/kfyy/cjj/jszl/200604/4002.html


C++中的set_new_handler函数

首先,namespace std中有如下定义:   Typedef void  (*new_handler)();         new_handler  set_new_handler(new...
  • wzxq123
  • wzxq123
  • 2016年05月25日 22:11
  • 2472

C++中的set_new_handler和new_handler 【未完待续】

转载自:c++中的set_new_handler和new_handler 详细可参考 《effective c++》第三版 条款49:了解new_handler的行为 相关链接: http://...
  • QQ1449301756
  • QQ1449301756
  • 2015年08月21日 16:19
  • 235

new_handler & set_new_handler

//the discriptions in msdn ,we can find this document in//website:http://msdn.microsoft.com/library/...
  • jiangredsheep
  • jiangredsheep
  • 2005年10月06日 13:47
  • 2558

c++中的set_new_handler和new_handler

原文链接:http://blog.csdn.net/liuxialong/article/details/6540605
  • lijun5635
  • lijun5635
  • 2014年04月06日 12:33
  • 606

operator new之错误处理函数new_handler

就像可以用atexit来注册main的退出处理函数一样,我们也可以用std::set_new_handler来注册operator new的错误处理函数。 当operator new无法满足某一...
  • u012421852
  • u012421852
  • 2016年07月10日 19:52
  • 334

了解new_handler的行为 set_new_handler

当operator new无法满足某一内存分配需求的时候,它会抛出异常。以前它会返回一个NULL指针,某些旧式编译器也还会那么做。现在也可以取得旧行为。     当operator new抛出异常来...
  • ProgramVAE
  • ProgramVAE
  • 2016年11月11日 22:24
  • 245

《STL源码剖析》中关于set_new_handler的理解

在阅读侯杰的《STL源码剖析》的时候,在书中的p45遇到一段代码,书中并没有给予set_new_handler详细的解释,查阅资料发现网上的解释并不是那么尽如人意,在查阅了《effective C++...
  • qq_14982047
  • qq_14982047
  • 2016年02月24日 17:50
  • 433

如何写出高效C++(定制new和delete和杂项讨论)

49.了解new-handle的行为 set_new_handler允许客户制定一个函数,在内存分配无法获得满足时被调用。 void outOfMem() {   std::cerr   std::a...
  • hello_bravo_
  • hello_bravo_
  • 2016年08月20日 10:05
  • 387

C++中new_handler

   在过去或者传统的模式中,当operator new不能满足一个内存分配请求的时候,它会返回一个空指针,但是现在如果operator new不能满足一个内存分配请求的时候,会抛出一个异常,这个时候...
  • gxj1680
  • gxj1680
  • 2009年12月24日 15:46
  • 866

《Effective C++》:条款49:了解new-handler的行为

C++内存是由程序员手动管理的,不像Java或.net有垃圾回收机制。C++内存管理主要是分配例程和归还例程(allocation and deallocation routines),即operat...
  • KangRoger
  • KangRoger
  • 2015年03月15日 21:03
  • 1428
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:c++中的set_new_handler和new_handler
举报原因:
原因补充:

(最多只允许输入30个字)