【52】写了placement new也要写placement delete

1、Widget* pw = new Widget; 调用了两个方法:第一个方法是operator new 负责分配内存;第二个方法是在分配的内存上构造Widget,即调用Widget的default构造方法。

2、那么问题来了,如果operator new分配内存成功,而Widget构造抛出异常,这种情况下,pw还没有赋值,这不就导致资源泄露了?

  不会造成资源泄露,因为C++运行系统保证调用delete释放资源。

3、正常的operator new 如下:

  void* operator new(std::size_t) throw (std::bad_alloc);

  正常的operator delete如下:

  void operator delete (void* rawMemory) throw(); // global作用域中的正常签名式

  void operator delete (void* rawMemory,std::size_t size) throw (); //class 作用域中典型的签名式

  除了上面的operator new,还有重载operator new,这些重载的new称为placement new,其中有一个比较特殊的,如下:

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

  当我们使用new在堆上申请内存时,编译器尝试找到一块内存。而上面的placement new是说,不要去找内存了,直接分配在pMemory上。

4、现在问题来了,存在多个重载的operator new,当我们调用一个重载的new,出现上面的情况,即构造Widget出现异常,运行时该调用哪个delete呢?

  做法是:调用一个与operator new 对应的operator delete,如果没有这样的delete,那么运行时系统什么也不做。这就必然导致资源泄漏。因此,为了避免这种情况下,用户必须保证operator new 存在与它对应的operator delete。

5、上面delete的调用,是说,使用operator new,构造对象时出现异常,运行期系统调用与之对应的operator delete。正常情况下,delete pw调用正常的delete。

6、为了避免漏掉相关的delete,可以建立一个父类,其它类继承,共享这些方法。需要注意的是,继承会导致掩盖同名称的成员,因此需要在子类中使用using XXX。

【52】写了placement new也要写placement delete

上一篇:可访问性级别的C# 修饰符


下一篇:C#递归题目代码