17370845950

c++ placement new是什么 c++在指定内存上创建对象【进阶】
placement new 是在指定内存地址上构造对象而不分配内存的机制,仅调用构造函数,适用于内存池、嵌入式系统、自定义容器等场景;需手动调用析构函数且不可用 delete。

placement new 是 C++ 中一种特殊的 new 表达式,它不分配内存,而是**在你已经准备好的、指定的内存地址上构造对象**。本质是“只调用构造函数”,跳过内存分配环节。

为什么需要 placement new?

常见于对内存控制要求严格的场景:

  • 内存池管理:预先分配一大块内存,反复复用,避免频繁堆分配开销
  • 嵌入式或实时系统:避免动态分配带来的不确定性(如碎片、延迟)
  • 自定义容器实现(如 std::vector 内部):先申请原始内存,再按需构造元素
  • 对象序列化/反序列化:把对象直接“还原”到某段已知内存中

基本语法和用法

标准形式为:new (address) Type(args...)

例如:

char buffer[sizeof(std::string)]; // 原始内存,未初始化
std::string* p = new (buffer) std::string("hello"); // 在 buffer 上构造 string

注意:buffer 必须足够大、对齐正确(alignas(std::string) char buffer[...] 更安全),且生命周期要长于对象。

必须手动调用析构函数

因为 placement new 没有分配内存,所以 delete 不能用 —— 它会尝试释放内存并调用析构函数,导致双重析构或崩溃。

正确做法是:显式调用析构函数,然后自行管理底层内存:

p->~string(); // 手动析构
// buffer 可以复用,或按原始方式释放(比如 free(buffer) 如果是 malloc 来的)

和普通 new 的关键区别

  • 普通 new = 分配内存 + 调用构造函数;placement new = 仅调用构造函数
  • 普通 delete = 调用析构函数 + 释放内存;placement new 对象只能手动析构,内存释放独立处理
  • placement new 不会抛出异常(除非构造函数自己 throw),但可重载全局或类内版本支持异常语义

不复杂但容易忽略。