placement new用于在指定内存构造对象,语法为new (ptr) Type(args),需手动调用析构函数并确保内存对齐与大小正确。

在c++中,placement new 是一种特殊的 new 表达式,用于在已分配的内存块上构造对象。它不会分配新的内存,而是在指定的内存地址处调用构造函数创建对象。这种机制常用于需要精确控制内存布局的场景,比如内存池、嵌入式系统、STL容器实现等。
placement new 的基本语法
placement new 的语法如下:
new (pointer_to_memory) Type(arguments);
其中 pointer_to_memory 是一个指向已分配内存的指针,Type 是要构造的对象类型,arguments 是传递给构造函数的参数(可选)。
使用步骤与示例
使用 placement new 通常分为三步:分配原始内存、构造对象、手动析构。
立即学习“C++免费学习笔记(深入)”;
1. 分配原始内存
可以使用 malloc、new[] 或栈上数组等方式获取一块原始内存。
2. 在指定内存位置构造对象
使用 placement new 在这块内存上构造对象。
3. 手动调用析构函数
由于 placement new 没有分配内存,所以不能使用 delete。必须显式调用析构函数清理对象状态,然后释放内存。
include <iostream>
class MyClass {
public:
MyClass(int val) : data(val) {
cout << “构造 MyClass(” << data << “)n”;
}
~MyClass() {
cout << “析构 MyClass(” << data << “)n”;
}
void print() const { cout << “data = ” << data << “n”; }
private:
int data;
};
int main() {
// 步骤1:分配原始内存
char buffer[sizeof(MyClass)]; // 栈上内存
// 步骤2:使用 placement new 构造对象
MyClass* obj = new (buffer) MyClass(42);
// 使用对象
obj->print();
// 步骤3:手动调用析构函数
obj->~MyClass();
return 0;
}
输出结果:
构造 MyClass(42)
data = 42
析构 MyClass(42)
注意事项
使用 placement new 时需特别注意以下几点:
- 确保目标内存足够大且对齐正确,否则会导致未定义行为。
- 必须显式调用析构函数,否则资源可能泄漏(如锁、文件句柄等)。
- 不要对 placement new 构造的对象使用 delete,因为它没有通过普通 new 分配内存。
- 可以结合 operator new 的重载实现自定义内存管理策略。
常见用途
placement new 常用于以下场景:
- 实现内存池或对象池,避免频繁分配/释放内存。
- 在共享内存或多线程环境中构造对象。
- STL 容器内部实现(如 vector 的扩容过程中使用 placement new 构造新元素)。
- 嵌入式系统中对特定地址映射硬件寄存器的对象构造。
基本上就这些。placement new 提供了对对象构造位置的精细控制,但需要开发者自行管理生命周期和内存安全。用好了高效,用错了容易出问题。


