placement new 是在已分配内存上构造对象的机制,语法为 new (ptr) Type(args),仅调用构造函数而不分配内存。常用于内存池、嵌入式系统和 STL 容器实现。使用时需确保内存对齐,手动调用析构函数,禁止使用 delete,重复构造前须先析构原对象。

placement new 是 c++ 中一种特殊的 new 表达式,用于在已经分配好的内存地址上构造对象。它不负责分配内存,只负责调用构造函数初始化指定位置的对象。
什么是 placement new
标准的 new 操作符会完成两件事:一是从堆中分配内存,二是调用构造函数初始化对象。而 placement new 只做第二件事——在给定的内存区域上构造对象。
它的语法形式如下:
new (pointer_to_memory) Type(arguments);
其中 pointer_to_memory 是一个已分配的、足够容纳该类型的内存块指针。
立即学习“C++免费学习笔记(深入)”;
常见用途和使用场景
placement new 常用于以下几种情况:
- 内存池或自定义内存管理:预先分配一大块内存,之后在需要时使用 placement new 构造对象,提高性能并减少频繁分配释放带来的开销。
- 嵌入式系统或实时系统:避免运行时动态分配内存,确保确定性行为。
- STL 容器实现:如 vector 在扩容时,会在新内存中用 placement new 构造元素。
基本使用示例
下面是一个简单的例子,展示如何使用 placement new:
#include iostream>
using Namespace std;
Struct MyClass {
int value;
MyClass(int v) : value(v) { cout ~MyClass() { cout };
int main() {
// 预先分配足够大的内存块
alignas(MyClass) char buffer[sizeof(MyClass)];
// 使用 placement new 在 buffer 上构造对象<br> MyClass* obj = new (buffer) MyClass(42);<br><br> // 手动调用析构函数(不会释放内存)<br> obj->~MyClass();<br><br> return 0;<br>
}
注意这里必须手动调用析构函数,因为 placement new 不涉及内存释放。真正的内存回收是在 buffer 超出作用域时自动完成的。
注意事项与陷阱
使用 placement new 时需特别小心:
- 必须确保内存对齐:使用 alignas 或 std::aligned_storage 等方式保证内存对齐正确,否则可能导致未定义行为。
- 必须手动调用析构函数:否则对象的资源清理逻辑不会执行。
- 不能用 delete 释放 placement new 构造的对象:delete 会尝试释放内存并调用析构函数,但这块内存不是用 new 分配的,会导致错误。
- 重复构造前应先析构:同一块内存多次使用 placement new 前,要确保旧对象已被正确析构。
基本上就这些。placement new 是个强大但容易误用的工具,适合底层开发和性能敏感场景,在普通应用代码中较少直接使用。理解它有助于深入掌握 C++ 对象生命周期和内存模型。