placement new 是在已分配内存上构造对象的机制,语法为 new (ptr) Type(args),常用于内存池、共享内存等需精确控制内存的场景。

在C++中,placement new 是一种特殊的 new 表达式,用于在已经分配好的内存地址上构造对象。它并不分配新的内存,而是调用构造函数将对象“放置”到指定的内存位置。这种机制常用于需要精确控制内存布局的场景,比如内存池、嵌入式系统、自定义容器或高性能编程。
什么是 placement new?
通常使用 new 操作符时,会完成两件事:一是分配内存,二是调用构造函数初始化对象。而 placement new 跳过了内存分配步骤,只执行构造函数调用。
其基本语法如下:
new (pointer_to_memory) Type(args);
其中 pointer_to_memory 是一个指向已分配内存的指针,Type 是要构造的对象类型,args 是传递给构造函数的参数。
立即学习“C++免费学习笔记(深入)”;
为什么需要 placement new?
某些场景下,我们希望对象在特定内存区域创建,而不是由系统自动分配。常见用途包括:
- 内存池管理:预先分配一大块内存,然后在其中多次使用 placement new 构造对象,避免频繁调用系统 malloc/new。
- 共享内存或硬件映射内存:在操作系统提供的固定地址空间构造对象。
- 性能优化:减少动态内存分配开销,提升效率。
- 实现容器类(如 std::vector):预留空间后逐步构造元素。
如何使用 placement new?
下面是一个简单示例,展示如何在栈上分配内存并用 placement new 构造对象:
#include <iostream>
using namespace std;
<p>class MyClass {
public:
int value;
MyClass(int v) : value(v) {
cout << "构造 MyClass(" << value << ")\n";
}
~MyClass() {
cout << "析构 MyClass(" << value << ")\n";
}
};</p><p>int main() {
// 在栈上分配足够存储 MyClass 的内存
alignas(MyClass) char buffer[sizeof(MyClass)];</p><pre class='brush:php;toolbar:false;'>// 使用 placement new 在 buffer 上构造对象
MyClass* obj = new (buffer) MyClass(42);
// 使用对象
cout << "obj->value = " << obj->value << "\n";
// 必须手动调用析构函数
obj->~MyClass();
return 0;}
输出结果:
构造 MyClass(42)obj->value = 42
析构 MyClass(42)
注意事项与常见问题
使用 placement new 时必须注意以下几点:
- 不会分配内存:确保传入的指针指向的是合法且足够大的内存区域。
- 必须手动调用析构函数:因为没有调用普通 new,所以 delete 无法触发析构。对象生命周期结束前应显式调用析构函数。
- 注意内存对齐:使用 alignas 和 sizeof 确保缓冲区满足类型对齐要求。
- 不要对同一内存重复使用 placement new:除非先调用析构函数,否则可能造成资源泄漏或未定义行为。
基本上就这些。placement new 提供了对对象构造过程的精细控制,是 C++ 高级编程中的重要工具,虽然不常用,但在需要高效或底层控制时非常关键。











