C++中的placement new是什么?C++在指定内存上创建对象【内存管理】

placement new 是 C++ 中不分配内存、仅在指定地址调用构造函数的特殊 new 表达式;用于对象池、嵌入式内存布局、容器实现等场景,需手动调用析构函数并管理内存。

placement new 是 C++ 中一种特殊的 new 表达式,它不分配内存,而是直接在你已经准备好的、指定的内存地址上构造对象。

placement new 的核心作用:只调用构造函数,不申请内存

普通 new 会做两件事:调用 operator new 分配内存,再在该内存上调用构造函数。而 placement new 跳过第一步,只执行第二步——在你给定的指针所指向的内存区域上,显式调用对象的构造函数。

典型写法:

void* buf = malloc(sizeof(MyClass));
MyClass* obj = new(buf) MyClass(); // 在 buf 指向的内存上构造对象

为什么需要 placement new?常见使用场景

它解决的是“控制对象诞生位置”的需求,典型用途包括:

  • 对象池或内存池管理:复用预分配的大块内存,避免频繁堆分配
  • 嵌入式或实时系统:内存布局严格受限,需把对象放在特定地址(如硬件寄存器映射区)
  • 自定义容器实现(如 std::vector 内部):先分配原始内存,后按需构造元素
  • 序列化/反序列化:把二进制数据直接“复活”为对象(需确保内存布局兼容且无虚函数等复杂特性)

必须手动调用析构函数,不能用 delete

因为 placement new 没有调用 operator new,所以不能用 delete 销毁对象——这会导致未定义行为。

正确做法是:显式调用析构函数,再按需释放底层内存(比如用 free() 或归还给内存池):

obj->~MyClass(); // 显式析构
free(buf); // 手动释放原始内存

注意:标准库已提供 placement new 声明,但不提供对应 placement delete

头文件 中声明了:

void* operator new(std::size_t, void* p) noexcept; // placement new 版本

这个函数只是返回传入的 p,不做任何内存分配。它没有对应的 operator delete 版本(因为不需要),所以也不存在 “placement delete” 这个东西——析构永远靠显式调用。

基本上就这些。用好 placement new 的关键是:心里清楚内存谁分配、谁释放,构造和析构必须严格配对,且避开编译器自动管理的那套流程。