placement new是在已分配内存上构造对象的C++机制,用于精细控制内存管理。它不分配新内存,而是接收现有地址并调用构造函数,需包含头文件。使用步骤包括:准备原始内存、用placement new构造对象、正常使用、手动调用析构函数、释放内存。示例中用alignas对齐的缓冲区存放Widget对象,在buffer上构造并析构。常见用途有内存池、实时系统、对象复用和STL容器实现。注意事项:必须显式调用析构函数,禁止对placement new指针使用delete,确保内存对齐与大小,重复使用前应先析构旧对象。例如用operator new分配内存,placement new构造,最后operator delete释放。正确配对构造与析构,自行管理内存,可安全高效使用placement new。

在C++中,placement new 是一种特殊的 new 表达式,允许你在已分配的内存上构造对象。它把内存分配和对象构造这两个过程解耦,适用于需要精细控制内存管理的场景,比如内存池、嵌入式系统或自定义容器。
什么是 placement new
placement new 并不分配新的内存,而是接收一个已存在的内存地址,在该位置调用对象的构造函数。它属于 C++ 标准库的一部分,定义在
最常用的语法是:
new (pointer_to_memory) Type(args)
立即学习“C++免费学习笔记(深入)”;
其中 pointer_to_memory 是一块足够容纳 Type 类型对象的原始内存。
使用步骤与示例
使用 placement new 通常包含以下几个步骤:
- 准备一块原始内存(可以用 malloc、new[]、栈数组等)
- 使用 placement new 在这块内存上构造对象
- 正常使用对象
- 手动调用析构函数清理对象状态
- 释放原始内存(如果动态分配)
示例代码:
#include#include // 必须包含 struct Widget { int value; Widget(int v) : value(v) { std::cout << "构造 Widget(" << value << ")\n"; } ~Widget() { std::cout << "析构 Widget(" << value << ")\n"; } };
int main() { // 分配原始内存(未构造对象) alignas(Widget) char buffer[sizeof(Widget)];
// 使用 placement new 构造对象 Widget* w = new (buffer) Widget(42); std::cout zuojiankuohaophpcnzuojiankuohaophpcn "w-youjiankuohaophpcnvalue = " zuojiankuohaophpcnzuojiankuohaophpcn w-youjiankuohaophpcnvalue zuojiankuohaophpcnzuojiankuohaophpcn "\n"; // 手动调用析构函数 w-youjiankuohaophpcn~Widget(); // 不需要 delete,因为内存不是用 new 分配的 return 0;}
常见用途
placement new 主要用于以下几种情况:
- 内存池管理:预先分配大块内存,用 placement new 在其中创建对象,避免频繁调用系统分配器
- 实时系统:避免运行时内存分配带来的不确定性延迟
- 对象复用:在相同内存位置反复构造/析构对象,减少分配开销
- 容器实现:STL 容器如 std::vector 内部就使用类似机制分离内存分配与构造
注意事项与陷阱
使用 placement new 需要格外小心:
- 必须显式调用析构函数,否则资源可能泄漏
- 不能对 placement new 返回的指针使用 delete,只能调用析构函数
- 确保内存对齐和大小足够,否则行为未定义
- 重复使用同一块内存前,应先析构旧对象
例如,用 operator new 分配原始内存:
void* mem = operator new(sizeof(Widget)); Widget* w = new (mem) Widget(100); // ... 使用 w->~Widget(); operator delete(mem); // 正确释放
基本上就这些。placement new 提供了底层控制力,但责任也更大。只要记得构造和析构成对出现,内存管理自己负责,就能安全使用。











