条款49: 了解new-handler的行为

operator new 和 operator delete只适合用来分配单一对象。array所用的内存由operator new[]分配出来,并由operator delete[] 释放。

1、了解new-handler的行为

当operator new无法满足某一内存分配需求时,它会先调用指定的错误处理函数(new-handler),然后抛出异常。可通过set_new_handler函数来设置new-handler。

namespace std {
typedef void (*new_handler)();
new_handler set_new_handler(new_handler p) throw();
}

new_handler是个typedef,定义出一个指针指向函数,该函数没有参数也不返回任何东西。

为当前class提供专属new-handler,只需为它提供自己的set_new_handler和operator new(分配内存)即可。

class Widget {
public:
static std::new_handler set_new_handler(std::new_handler p) throw();
static void* operator new(std::size_t size) throw(std::bad_alloc);
private:
static std::new_handler currentHandler;
}; std::new_handler Widget::currentHandler = ; // 注意此处要返回之前使用的new_handler
std::new_handler Widget::set_new_handler(std::new_handler p) throw() {
std::new_handler oldHandler = currentHandler;
currentHandler = p;
return oldHandler;
} void* Widget::operator new(std::size_t size) throw(std::bad_alloc) {
NewHandlerHolder h(std::set_new_handler(currentHandler));
return ::operator new(size);
} // 为确保原本的new-handler能够被重新安装回去, 运用资源管理对象
class NewHandlerHolder {
public:
explicit NewHandlerHolder(std::new_handler nh): handler(nh){}
~NewHandlerHolder(){
std::set_new_handler(handler);
}
private:
std::new_handler handler;
// 防止copy
NewHandlerHolder(const NewHandlerHolder&);
NewHandlerHolder& operator=(const NewHandlerHolder&);
};

使用nothrow new只能保证operator new不抛出异常,若分配失败则返回NULL,不保证new (std::nothrow) Widget这样的表达式绝不导致异常,因为Widget构造函数中可能继续new内存出现异常

Widget *p = new (std::nothrow) Widget;

条款50:了解new和delete的合理替换时机 <pass>

条款51:编写new和delete时需固守常规

1、operator new应该内含一个无穷循环, 并在其中尝试分配内存,如果它无法满足内存需求,就该调用new-handler.

2、operator delete应该在收到null指针时不做任何事

class Base {
public:
static void* operator new(std::size_t size) throw(std::bad_alloc) ;
static void operator delete(void *rawMemory, std::size_t size) throw();
}; /*
operator new成员函数会被derived class继承,而定制内存管理器的一个常见理由是为某特定class的对象
分配行为提供最优化,即针对Base类设计的operator new其行为可能只为大小刚好为sizeof(Base)的对象而设计
*/ void* Base::operator new(std::size_t size) throw(std::bad_alloc) {
if (size != sizeof(Base)) {
return ::operator new(size);
}
// 专为Base类定制的行为
// ...
} void Base::operator delete(void *rawMemory, std::size_t size) throw() {
if (rawMemory == ) {
return;
}
// 与上述::operator new对应
if (size != sizeof(Base)) {
::operator delete(rawMemory);
return;
}
// 释放rawMemory所指的内存
return;
}

条款52:写了placement new也要写placement delete

placement new:  多了pMemory参数

void* operator new(std::size_t size, void* pMemory) throw();

如果一个带额外参数的operator new 没有带相同额外参数的对应版operator delete,那么当new的内存分配动作需要取消并恢复原样时就没有任何operator delete会被调用,导致内存泄漏。

placement delete只有在伴随placement new调用而触发的构造函数出现异常时才会被调用,delete某指针时不会导致调用placement delete。

class Widget {
public:
static void* operator new(std::size_t size, std::ostream &logStream) throw (std::bad_alloc);
static void operator delete(void *pMemory) throw();
static void operator delete(void *pMemory, std::ostream &logStream) throw();
}; // 若引发Widget构造函数抛出异常,会自动调用placement delete
Widget *pw = new (std::cerr) Widget;
// 调用正常的operator delete
delete pw;

缺省情况下c++在global作用域内提供以下形式的operator new:

void* operator new(std::size_t) throw(std::bad_alloc); // normal new
void* operator new(std::size_t, void*) throw(); // placement new
void* operator new(std::size_t, const std::nothrow_t&) throw();

如果在class内声明任何operator new,会隐藏上述这些标准形式。为避免出现这种情况,可以建立一个Base class,内含所有正常形式的new和delete,凡是想以自定义形式扩充标准形式的客户,可利用继承机制及using 声明式取得标准形式。

【effective c++】定制new和delete的更多相关文章

  1. Effective C++ —— 定制new和delete(八)

    STL容器所使用的heap内存是由容器所拥有的分配器对象管理,不是被new和delete直接管理.本章并不讨论STL分配器. 条款49 : 了解new-handler的行为 当operator new ...

  2. 高效C++:定制new和delete

    内存的申请和释放,C++从语言级别提供了new和delete关键字,因此需要了解和熟悉其中的过程. 了解new-handler的行为 set_new_handler可以指定一个函数,当申请内存失败时调 ...

  3. Effective C++: 08定制new和delete

    49:了解new-handler的行为 当operator new无法满足某一内存分配需求时,它会抛出异常(以前会返回一个null).在抛出异常之前,它会调用一个客户指定的错误处理函数,也就是所谓的n ...

  4. 《Effective C++》定制new和delete:条款49-条款52

    条款49:了解new-handler的行为 当operator new无法分配出内存会抛出异常std::bad_alloc 抛出异常前会反复调用用户自定义的new-handler函数直至成功分配内存 ...

  5. 八、定制new和delete

    条款49:了解new-handler的行为 new异常会发生什么事? 在旧式的编译器中,operator new分配内存失败的时候,会返回一个null指针.而现在则是会抛出一个异常. 而在抛出这个异常 ...

  6. 《Effective C++》读书摘要

    http://www.cnblogs.com/fanzhidongyzby/archive/2012/11/18/2775603.html 1.让自己习惯C++ 条款01:视C++为一个语言联邦 条款 ...

  7. 《Effective C++》 阅读小结 (笔记)

    A person who is virtuous is also courteous. "有德者必知礼" 书本介绍:<Effective C++:改善程序与设计的55个具体做 ...

  8. C++ delete 和 delete []

    C++ delete 和 delete [] 简单结论: new delete new [] delete []   文章 : 对 delete [] 的声明 void operator delete ...

  9. effective c++ 思维导图

    历时两个多月的时间,终于把effective c++又复习了一遍,比较慢,看的是英文版,之前看的时候做过一些笔记,但不够详细,这次笔者是从头到尾的翻译了一遍,加了一些标题,先记录到word里面,然后发 ...

随机推荐

  1. 前端上传控件plupload总结

    plupload是一个单图和多图上传控件: 属性和方法介绍,参考以下博客: https://www.cnblogs.com/2050/p/3913184.html 这里直接贴出JS代码,细到爆的注释, ...

  2. 优化mysql查询

    mysql提供了一个特别的explain语句,用来分析查询语句的性能 : explain select ... 1.在所有用于where,order by,group by的列上添加索引 创建索引 添 ...

  3. 【php】关于尾部去除和分号问题

    One thing to remember is, if you decide to omit the closing PHP tag, then the last line of the file ...

  4. day23 02 组合(继续人狗大战游戏)

    day23 02 组合(继续人狗大战游戏) 面向对象的三大特性:继承 多态 封装 先讲解一下组合 组合:一个对象的属性值是另一个类的对象:对象.属性.属性(一般有两个点) 继续扩展day22 01里面 ...

  5. JavaScript正则表达式-或字符

    或字符是一个单竖线“|”,表示字符串只要匹配“|”连接的多个表达式中的任意一个. /ab|ac|bc/表示字符串匹配ab,或者匹配ac,或者bc

  6. luogu3629 [APIO2010]巡逻

    创造一个环出来,可以让环上的边都只访问一次. 对于 \(k=1\),答案就是树的直径两边连起来. 倘若 \(k=2\),那就先按照 \(k=1\) 的求一遍,然后我们发现,如果第二条加的边构成的环和第 ...

  7. 对 Servlet 的改进

    通过上一篇博客:Servlet 的详解 http://www.cnblogs.com/ysocean/p/6912191.html,我们大致知道了 Servlet 的基本用法.但是稍微分析一下 Ser ...

  8. jsonp实现跨域访问json数据

    前台js function init() { $.ajax({ url: 'http://localhost:8012/index.json', dataType: "jsonp" ...

  9. 此坑待填 离散化思想和凸包 UVA - 10173 Smallest Bounding Rectangle

    Smallest Bounding Rectangle Given the Cartesian coordinates of n(>0)2-dimensional points, write a ...

  10. iRule Event Order - HTTPSv7

    v