placement delete是operator delete的重载形式,与placement new配对使用,在构造函数抛出异常时自动调用以释放已分配内存,确保异常安全;它由编译器在placement new失败时触发,不会被显式调用,主要用于防止内存泄漏,其典型形式为void operator delete(void, void)。

在C++中,placement delete是一个与placement new配对使用的特殊机制。它并不像普通delete那样释放内存,而是在placement new抛出异常时自动调用,用于清理已被分配但对象构造失败的内存。理解placement delete的关键在于明白它通常不会被程序员显式调用,而是作为异常安全机制的一部分由编译器自动触发。
什么是placement delete?
placement delete是operator delete的一种重载形式,它与placement new成对出现。当你使用placement new在指定内存位置构造对象时,如果构造函数抛出异常,系统需要释放之前分配的内存,这时就会调用对应的placement delete函数。
标准形式如下:
void operator delete(void* ptr, std::size_t size); void operator delete(void* ptr, void* place); // 常见的placement delete
placement delete的调用时机
placement delete不是用来手动释放对象的。它的主要作用是异常清理。当以下情况发生时,placement delete会被自动调用:
立即学习“C++免费学习笔记(深入)”;
- 使用placement new前,先分配了原始内存
- 在构造函数中抛出了异常
- 此时系统会调用匹配的placement delete来释放已分配的内存
如果没有定义对应的placement delete,只会导致内存泄露,而不会调用普通delete。
自定义placement new和placement delete示例
下面是一个完整的例子,展示如何定义和使用placement new/delete:
#include#include struct Test { int value; Test(int v) : value(v) { if (v == 42) throw std::runtime_error("bad value"); std::cout << "Constructed: " << value << "\n"; } };
// 自定义placement new void operator new(std::size_t size, void ptr) { std::cout << "Custom placement new called\n"; return ptr; }
// 对应的placement delete(仅在构造异常时调用) void operator delete(void ptr, void place) { std::cout << "Custom placement delete called for " << ptr << "\n"; // 注意:这里不真正释放内存,只是清理记录 }
int main() { char buffer[sizeof(Test)];
try { Test* t = new(buffer) Test(42); // 构造函数抛异常 } catch (...) { std::cout zuojiankuohaophpcnzuojiankuohaophpcn "Exception caught\n"; } return 0;}
输出结果:
Custom placement new called Exception caught Custom placement delete called for [地址]
可以看到,虽然我们没有手动调用delete,但placement delete仍然被执行了。
常见误区与注意事项
很多人误以为可以像这样手动调用placement delete:
operator delete(ptr, buffer);这是错误的做法。正确的做法是先调用析构函数,然后不再需要做任何事,因为placement new使用的内存通常由你管理(比如栈上数组),不需要也不应该通过delete释放。
正确销毁placement new创建的对象方式是:
t->~Test(); // 显式调用析构 // 不再需要其他操作
基本上就这些。placement delete的核心价值在于异常安全,确保即使构造失败也不会造成资源泄露。它不是供日常使用的工具,而是RAII和异常处理机制中的重要一环。











