Loading

《Effective C++》定制new和delete

Item49:了解new_handler的行为
当operator new抛出异常以反映出一个未获得满足的内存需求之前,它会先调用一个用户制定的错误处理函数,一个所谓的new-handler,为了制定这个“用以处理内存不足”的函数,用户必须调用set-new-handler,那是声明于中的一个标准程序库函数:

namespace std{
    typedef void(*new_handler)();
    new_handler set_new_handler(new_handler p) throw();//这里是nothrow吗?书中是throw
}

new_handler是个typedef,定义出一个指针指向一个函数,这个函数没有参数也不反悔任何东西,set_new_handler则是获得一个new_handler并且返回一个new_handler的函数,set_new_handler声明式尾端的”throw()”是一份异常明细,表明该函数不抛出任何异常(是不是nothrow?)
set_new_handler的参数也是个指针,指向operator new无法分配足够内存时候该被调用的函数,其返回值也是个指针,指向set_new_handler被调用前正在执行(但马上就要被替换)的那个new_handler函数,例如:

void outofMem()
{
    std::cerr<<"无法分配内存";
    std::abort();
}
int main()
{
std::set_new_handler(outofMem);
int * pBigDataArray=new int[100000000L];
}

一个涉及良好的new-handler函数必须做到以下事情:

  1. 让更多内存可被使用,实现策略是,程序一开始执行就分配一大块内存,而后当new_handler第一次被调用,将它们释还给程序使用。
  2. 安装另一个new_handler,如果这个new_handler无法获取更多可用内存,它可以自己调用set_new_handler来将其它函数来替换自己。
  3. 删除new_handler,也就是将null指针传给set_new_handler,一旦没有安装任何new_handler,operator new会在内存分配不成功时候抛出异常。
  4. 抛出bad_alloc异常,这样的异常不会被operator new捕捉,因此会传播到内存需求处。
  5. 不反悔,通常调用abort()或者exit()

    nothrow new是一个颇为局限的工具,因为它只适用于内存分配,后继的构造函数调用还是可能抛出异常。


Item50:了解new和delete的合理替换时机
在什么情况下会需要替换new和delete?

  1. 用来检测运行上的错误。
  2. 为了提高性能。
  3. 为了收集使用上的统计数据。
  4. 为了增加分配和归还的速度。
  5. 为了降低缺省内存管理器带来的空间额外开销。
  6. 为了尼姆缺省分配器中的非最佳齐位。
  7. 为了将相关对象成簇集中。
  8. 为了获得非传统行为。

Item51:编写new和delete时需固守成规
总结:①operator new应该内含一个无穷循环,并且其中尝试分配内存,如果它无法满足内存需求,就应该调用new_handler,它也应该有能力处理0字节申请。
②operator delete应该在收到null指针时候不做任何事情

posted @ 2017-04-11 15:35  李正浩  阅读(102)  评论(0编辑  收藏  举报