为什么 std::make_shared 无法编译带有已删除运算符 new 的类型?

Why doesn't std::make_shared fail to compile for type with deleted operator new?

本文关键字:运算符 删除 new 类型 编译 make std shared 为什么      更新时间:2023-10-16

>我正在尝试编写堆禁止类型,即在堆分配的内存上不可构造的类型。通过删除运算符 new 和放置 new,我认为我会实现这一目标。但是使用std::make_shared创建共享指针仍然可以编译。

为什么删除新运算符时std::make_shared<A>()不会编译失败?

#include <memory>
class A {
public:
void* operator new(size_t) = delete;
void* operator new(size_t, void*) = delete;
void* operator new [] (size_t) = delete;
};
// Regular new fails
A* a1 = new A();
// Placement new fails
void* pv = std::malloc(sizeof(A));
A* a2 = new (pv) A();
// make_shared works
std::shared_ptr<A> a3 = std::make_shared<A>();
std::make_shared

是根据::​new (pv) T(std​::​forward<Args>(args)...)指定的。

[util.smartptr.shared.create]

2效果:分配适合 T 型对象的内存 并通过放置在该内存中构造一个对象 新表达::​new (pv) T(std​::​forward<Args>(args)...).这 模板allocate_shared使用 A 的副本来分配内存。如果 引发异常,函数不起作用。

分配的内存通常用于控制块,而不是直接用于new T。然后通过放置 new 构造对象,但新表达式完全有资格使用全局放置operator new,而不是任何特定于类的放置。

[新品]

9 如果新表达式以一元运算符开头​::则 分配函数的名称在全局范围内查找。 否则,如果分配的类型是类类型 T 或其数组, 分配函数的名称在 T 的作用域中查找。如果这个 查找找不到名称,或者分配的类型不是类 类型,则在全局范围内查找分配函数的名称。

这两种用法将完全绕过自定义删除的运算符。