placement new用于在已分配内存上构造对象,不分配内存仅调用构造函数,需手动调用析构函数且不能使用delete,常见于内存池、STL容器等需精细内存控制的场景。

在C++中,placement new 是一种特殊的 new 表达式,用于在已分配的原始内存上构造对象。它不负责分配内存,只负责调用构造函数,在指定地址上初始化对象。这种机制常用于需要精细控制内存管理的场景,如自定义内存池、嵌入式系统或STL容器的实现。
placement new 的基本语法与作用
标准的 new 操作符会完成两件事:一是分配内存,二是调用构造函数。而 placement new 只做第二步——在给定的内存位置构造对象。
其基本语法如下:
void* buffer = malloc(sizeof(MyClass)); MyClass* obj = new (buffer) MyClass(); // 在 buffer 所指向的内存上构造对象
这里的 new (buffer) 就是 placement new,括号中的 buffer 是一个已分配的内存地址。表达式会在该地址上调用 MyClass 的构造函数。
立即学习“C++免费学习笔记(深入)”;
常见的 placement new 形式
除了最简单的指针形式,C++ 还支持多种 placement new 重载方式:
-
标准 placement new:接收一个
void*参数,位于头文件中。 - 用户自定义 placement new:类可以重载 placement new 操作符,接受额外参数,例如:
void* operator new(size_t, std::align_val_t, const char* file, int line);
// 调用时可传入调试信息
new (std::align_val_t{16}, __FILE__, __LINE__) MyClass;
这在调试内存分配或跟踪对象创建位置时非常有用。
使用 placement new 的注意事项
虽然 placement new 提供了灵活的对象构造方式,但使用时必须格外小心:
- 手动调用析构函数:由于对象不是通过普通 new 创建的,delete 无法自动清理。必须显式调用析构函数: obj->~MyClass();
-
确保内存对齐:提供的缓冲区必须满足目标类型的对齐要求。例如,double 类型通常要求 8 字节对齐。使用
aligned_storage或std::aligned_alloc可避免问题。 - 避免重复构造:在同一块内存上多次使用 placement new 而未先析构,会导致未定义行为。
- 不要对 placement new 返回的对象使用 delete:delete 会尝试释放内存并调用析构,但这块内存并非由堆分配器分配,调用 delete 会导致崩溃或未定义行为。
典型应用场景
placement new 常见于以下情况:
- 内存池管理:预先分配一大块内存,然后在其中多次使用 placement new 构造对象,提升性能。
- STL 容器实现:vector 在扩容时,常使用 placement new 在新内存中构造元素。
- 共享内存或 mmap 映射区域:在操作系统映射的固定地址上构造 C++ 对象。
基本上就这些。placement new 是强大但危险的工具,掌握它的前提是理解对象生命周期与内存管理的分离。正确使用能提升效率,滥用则容易引发内存错误。











