placement new用于在指定内存地址构造对象,不分配内存,需手动调用析构函数,常用于内存池、STL容器等高性能场景。

在C++中,placement new 是一种特殊的 new 表达式,用于在已分配的内存块上构造对象。它不负责分配内存,而是将对象构造过程“放置”到指定的内存地址中。这种机制在需要精细控制内存管理的场景中非常有用,比如自定义内存池、嵌入式系统或STL容器的实现。
placement new 的基本语法
标准的 new 表达式会做两件事:调用 operator new 分配内存,然后在该内存上调用构造函数。而 placement new 只做第二步——调用构造函数。
其基本语法如下:
ClassName* ptr = new (address) ClassName(args);其中 address 是一个已经分配好的内存地址(通常是指针),ClassName(args) 是要构造的对象及其参数。
立即学习“C++免费学习笔记(深入)”;
常见用途与示例
placement new 最常见的用途包括:
- 在栈内存上构造对象
- 实现对象池或内存池
- 避免频繁的堆分配,提升性能
- 用于 STL 容器内部实现(如 vector 在预留空间中构造元素)
下面是一个简单示例,展示如何使用 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 上构造对象
MyClass* obj = new (buffer) MyClass(42);
cout zuojiankuohaophpcnzuojiankuohaophpcn "obj-youjiankuohaophpcnvalue = " zuojiankuohaophpcnzuojiankuohaophpcn obj-youjiankuohaophpcnvalue zuojiankuohaophpcnzuojiankuohaophpcn endl;
// 必须显式调用析构函数
obj-youjiankuohaophpcn~MyClass();
return 0;
}
输出结果:
构造函数被调用,value = 42obj->value = 42
析构函数被调用
注意事项与关键点
使用 placement new 时必须注意以下几点:
- 不分配内存:placement new 不会申请新内存,传入的地址必须有效且足够容纳对象。
- 必须手动调用析构函数:由于没有通过普通 new 创建对象,delete 无法使用。必须显式调用析构函数来清理资源。
- 内存对齐:确保缓冲区正确对齐。使用 alignas 可以帮助处理对齐问题。
- 避免重复构造:不要在同一个内存区域多次使用 placement new 而不先析构,否则会导致未定义行为。
也可以定义自己的 placement new 版本,只需重载 operator new 并接受额外参数。例如:
void* operator new(size_t size, void* ptr) {return ptr;
}
// 这是标准库内置的版本,通常不需要自己写
基本上就这些。placement new 提供了对对象构造位置的精确控制,适合高性能或资源受限环境,但使用时要格外小心内存生命周期管理。











