placement new 是C++中用于在已分配内存上构造对象的机制,不分配新内存,仅调用构造函数。其语法为 new (ptr) Type(args);,适用于内存池、STL容器实现等需精细控制对象生命周期的场景。使用步骤包括:准备足够且对齐的原始内存,用 placement new 构造对象,手动调用析构函数,自行管理内存释放。示例中利用 alignas 确保栈上缓冲区对齐,并在使用后显式调用析构函数。常见应用有内存池、共享内存通信和嵌入式系统。注意事项包括避免重复构造、确保正确对齐与大小、防止栈溢出,推荐结合 RAII 如自定义删除器的 unique_ptr 来安全管理资源。

在C++中,placement new 是一种特殊的 new 表达式,允许你在已分配的原始内存上构造对象。它不负责分配内存,只负责调用构造函数,在指定地址初始化对象。这个特性在实现高性能内存池、自定义容器、嵌入式系统或需要精细控制对象生命周期的场景中非常有用。
什么是 placement new?
placement new 并不会分配新的内存空间,而是使用你提供的内存地址来构造对象。语法如下:
new (pointer_to_memory) Type(arguments);其中 pointer_to_memory 是一个指向已分配内存的指针,Type 是要构造的对象类型。
如何使用 placement new 构造对象?
下面是使用 placement new 在指定内存位置构造对象的基本步骤:
立即学习“C++免费学习笔记(深入)”;
- 准备一块足够大的原始内存(可以是栈上数组、堆上分配的内存或 mmap 映射的内存)
- 使用 placement new 在这块内存上调用构造函数
- 使用完对象后,手动调用析构函数
- 不通过 delete 释放对象,而是自行管理内存的释放
示例代码:
#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 构造对象
MyClass* obj = new (buffer) MyClass(42);
cout zuojiankuohaophpcnzuojiankuohaophpcn "obj-youjiankuohaophpcnvalue = " zuojiankuohaophpcnzuojiankuohaophpcn obj-youjiankuohaophpcnvalue zuojiankuohaophpcnzuojiankuohaophpcn endl;
// 手动调用析构函数
obj-youjiankuohaophpcn~MyClass();
return 0;
}
输出:
obj->value = 42
析构: 42
注意:alignas 确保 buffer 的对齐满足 MyClass 的要求,避免未对齐访问错误。
常见应用场景
placement new 常用于以下几种高级场景:
- 内存池管理:预先分配一大块内存,按需使用 placement new 构造对象,提升性能
- STL 容器实现:vector 或 deque 内部使用未初始化内存缓存元素,插入时才构造
- 共享内存通信:多个进程共享同一块内存区域,通过 placement new 初始化对象
- 嵌入式系统:在特定硬件地址构造设备驱动对象
注意事项与最佳实践
使用 placement new 需要格外小心,否则容易引发资源泄漏或未定义行为:
- 必须手动调用析构函数,不能依赖自动析构(除非封装在 RAII 类中)
- 确保内存大小足够且正确对齐
- 不要对同一块内存重复使用 placement new 而不先析构原对象
- 避免在栈上分配大对象缓冲区,可能造成栈溢出
- 考虑使用 std::aligned_storage 或 std::byte[align_val_t] 来管理原始内存
现代 C++ 中,可结合 std::unique_ptr 和自定义删除器来安全管理 placement new 对象:
auto deleter = [](MyClass* p) {p->~MyClass();
};
unique_ptr
基本上就这些。placement new 提供了对对象构造过程的底层控制能力,但需要开发者承担更多责任。只要谨慎使用,并配合 RAII 思想,就能在性能和安全性之间取得良好平衡。









