1. operator new、operator delete与new、delete操作符的区别:

  • operator new的作用类似于malloc,负责分配内存;operator delete的作用类似于free,负责释放内存
  • new操作符包括两个步骤;调用适当的operator new版本分配内存;在分配的内存上调用构造函数
  • delete操作符包括两个步骤:在内存上调用析构函数;调用适当的operator delete版本释放内存
  • operator new和operator delete函数可以重载、重写(名字遮掩),而new/delete操作符不可

2. 标准库提供的operator new/delete(C++11)

标准库提供的global operator new,在全局名字空间中定义:
    void* operator new (std::size_t size); //抛出异常的版本
void* operator new (std::size_t size, const std::nothrow_t& nothrow_value) noexcept; //不抛出异常的版本,内存不足则返回nullptr
void* operator new (std::size_t size, void* ptr) noexcept; //placement new,仅仅返回ptr值,未作内存分配

对应的global operator delete:

    void operator delete (void* ptr) noexcept;
void operator delete (void* ptr, const std::nothrow_t& nothrow_constant) noexcept;
void operator delete (void* ptr, void* voidptr2) noexcept;//本函数未做任何事情

3. operator new与operator delete:层次关系、版本、对应关系与new handler

  • 层次关系:C++标准中仅建议在两种“作用域”中定义operator new/delete,即在类中作为类的static成员或在global作用域中。在使用Class1 *pc = new Class1;的形式创建动态对象时,new操作符首先在当前类(Class1)中查找operator new的相应版本,若未找到,则会到基类中递归查找,若还未找到,则会到global作用域中查找。查找过程中使用“名字遮掩规则”
  • 版本:
    1. 标准版:标准版的operator new只有一个size_t类型的参数;标准版的operator delete只有 void*类型参数;另外,operator delete还有一个特别成员版:有void *类型参数和size_t类型参数(只可在class中定义)。若类中同时定义了这两种operator delete,则优先使用前者。
    2. 非标准版:Effective C++中称之为placement版,并指出其多义性,而C++标准中将placement语义留给了上述提到标准库的第三个函数。非标准版的operator new可以有除了第一个size_t参数以外的其他参数,原型形式为:void* operator new(std::size_t size, type1 arg1, type2 arg2...); 通过new操作符间接调用的形式为: new(arg1, arg2, ...)ClassX; 非标准版的operator delete除了有第一个void*类型的参数还有额外的参数,其原型形式为:void* operator delete(void *ptr, type1 arg1, type2 arg2 ...); 非标准版的operator delete会在相应的operator new调用后对象的构造函数发生异常时由编译器调用。
    3. 注:关于operator delete的“特别成员版”。若类中同时有标准形式的operator delete作为成员,则delete操作符优先使用标准形式的operator delete,即时在异常抛出(下文会解释这是什么异常情况)时也是如此。另外,非标准版的operator new并没有与之参数对应的“特别成员版”
不同形式的operator new与operator delete:
    void* operator new (std::size_t size); //全局标准版operator new
void operator delete(void* ptr);//全局标准版operator delete
class A{
public:
static void operator new(size_t size);//类专属标准版operator new
static void operator delete(void* ptr, size_t size) noexcept;//类特别成员版operator delete
static void operator delete(void* ptr) noexcept;//类专属operator delete
};
  • 对应关系:
    1. 标准版:void* operator new(std::size_t size)可与两种operator delete对应,即void operator delete(void* ptr)或void operator delete(void*ptr, size_t)。且无论在正常情况下,还是异常情况下,C++优先使用第一种operator delete
    2. 非标准版:void* operator new(std::size_t size, ...)与void operator delete(void* ptr, ...)对应。未发生异常时,C++编译器调用operator new(标准版或非标准版依据new操作符是否传递额外参数而定)分配内存,调用标准版的operator delete(无论该内存是否通过标准版的operator new)释放内存。异常发生时(operator new成功调用,但调用对象构造函数时发生异常),编译器会调用与operator new参数对应的operator delete版本,若未找到,则不会调用其他形式的operator delete
  • operator new与new handler。编写自定义的operator new,在发生异常或内存不足时,应正确调用已设置的new handler

4. 自定义operator new与operator delete

  • 可以重新定义标准库提供的非placement版本的operator new和operator delete
  • 建议在global和class作用域中重写或重载自定义的operator new和operator delete

5. operator new/delete与类层次

  • 类的operator new成员通常仅负责当前类的内存分配,并不负责子类的内存非配
  • 类的析构函数应该是virtual的,否则Base *pb = new Derived; delete pb;将不能传递正确的size_t参数给子类的operator delete成员

6. operator new/delete与operator new[]/delete[]

  • 以上规则均适用于operator new[], operator delete[],仅仅是函数名字不同

参考:

C++中的new与delete总结的更多相关文章

  1. Oracle中的Truncate和Delete语句

    Oracle中的Truncate和Delete语句   首先讲一下,truncate命令:   语法:TRUNCATE  TABLE  table; 表格里的数据被清空,存储空间被释放. 运行后会自动 ...

  2. oracle 中删除表 drop delete truncate

    oracle 中删除表 drop delete truncate   相同点,使用drop delete truncate 都会删除表中的内容 drop table 表名 delete from 表名 ...

  3. Hibernate Session中的save(),update(),delete(),saveOrUpdate() 细粒度分析

    Hibernate在对资料库进行操作之前,必须先取得Session实例,相当于JDBC在对资料库操作之前,必须先取得Connection实例, Session是Hibernate操作的基础,它不是设计 ...

  4. C++中的new与delete

    C++中对象数组创建时,主要注意的点有:虚函数和带参数的构造函数,当出现虚函数时,对象数组中如通过父对象指向子对象,因为需要做Slice,析构时会造成指针错误引发内存泄露.测试程序如下: class ...

  5. sql server 中删除表中数据truncate和delete的区别(转载自.net学习网)

    我们都知道truncate table可以用来删除整个表的内容,它与delete后面不跟where条件的效果是一样.但除此之外,我们还清楚它们之间有其它的区别吗?本章我们将一起讨论truncate与d ...

  6. Javascript 中的 in, hasOwnProperty, delete, for/in

    in 运算符 判断对象是否拥有某一属性只要对象拥有该属性,就会返回true,否则false var point = { x:1, y:1 };alert( 'x' in point );  //tru ...

  7. 含有自增序列的表中使用truncate与delete的不同结果

    一个含有自增序列的表,使用delete跟truncate之后会有什么不同结果呢? 大概说一下,使用truncate,表中的记录全部被清除,如果向表中插入数据,那么数据的排序是从1开始的. 如果使用的是 ...

  8. ASP.NET MVC 学习8、Controller中的Detail和Delete方法

    参考:http://www.asp.net/mvc/tutorials/mvc-4/getting-started-with-aspnet-mvc4/examining-the-details-and ...

  9. SharePoint 2010中列表Add和Delete事件解析

    转:http://winsystem.ctocio.com.cn/26/11400026_2.shtml [IT专家网独家撰稿]SharePoint 2010与以前的版本相比,天翻地覆的变化并不为过. ...

随机推荐

  1. Red hat Linux(Centos 5/6)安装R语言

    Red hat Linux(Centos 5/6)安装R语言1 wget http://cran.rstudio.com/src/base/R-3/R-3.0.2.tar.gz2 tar xzvf R ...

  2. Grunt 初体验

    对于没有接触过类似自动化工具的朋友,对 grunt 也许只是停留在听过阶段,而并没有真正的使用过.今天就从最初级的教程说起.在开始教程之前,需要先确保你已经安装了 node. 下面就开始来讲解 gru ...

  3. 解锁windowsphone设备遇到的错误:检查Miscrosoft账户凭据、请重新注册 0x80004005 解决方案

    本篇文章主要讲在解锁windowsphone设备时遇到的错误 Error1:登录windowsphone开发人员中心时出错,请检查您的miscrosoft账户凭据 Error2:注册您的手机时出现未知 ...

  4. BZOJ_1022_[SHOI2008]_小约翰的游戏John_(博弈论_反Nim游戏)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1022 反Nim游戏裸题.详见论文<组合游戏略述——浅谈SG游戏的若干拓展及变形>. ...

  5. SharePoint 2010 自定义 字段 类型--------省市区联动

    转:http://www.cnblogs.com/sp007/p/3384310.html 最近有几个朋友问到了有关自定义字段类型的问题,为了让更多的人了解自定义字段类型的方法,特写一篇博客与大家分享 ...

  6. 提示35. 怎样实现OfTypeOnly<TEntity>()这样的写法

    提示35. 怎样实现OfTypeOnly<TEntity>()这样的写法 如果你编写这样LINQ to Entities查询: 1 var results = from c in ctx. ...

  7. Android 退出提示框 代码

    转自:http://hi.baidu.com/ittdt/item/d932cf37f486f886c3cf29ea new AlertDialog.Builder(MainEngine.contex ...

  8. ln (link)命令

    ln是linux中又一个非常重要命令,它的功能是为某一个文件在另外一个位置建立一个同步的链接.当我们需要在不同的目录,用到相同的文件时,我们不需要在每一个需要的目录下都放一个必须相同的文件,我们只要在 ...

  9. C# 检测机器是否有声卡设备

    有时候我们的程序需要进行音频的播放,则我们首先需要判断机器是否有声卡能够进行音频的播放.在网上找了一下没有发现太多关于如何检机器是否有声卡的例子.我在看了一些文档后自己写了一个小测试程序,如果机器装有 ...

  10. CDR绘制绚丽五角星※※

    CDR绘制绚丽五角星 1.绘制一个五角星,在多边形工具下拉的第二个就是 2.选中五角星,点击颜色即可.给五角星加上颜色 3.用立体化工具进行延伸. 4.点击图形中心向下拉. 看到了中间的一个长方条了没 ...