operator delete, operator delete[]
定义于头文件 <new> |
||
可替换的常规解除分配函数 |
||
(1) | ||
void operator delete ( void* ptr ) throw(); |
(C++11 前) | |
void operator delete ( void* ptr ) noexcept; |
(C++11 起) | |
(2) | ||
void operator delete[]( void* ptr ) throw(); |
(C++11 前) | |
void operator delete[]( void* ptr ) noexcept; |
(C++11 起) | |
void operator delete ( void* ptr, std::align_val_t al ) noexcept; |
(3) | (C++17 起) |
void operator delete[]( void* ptr, std::align_val_t al ) noexcept; |
(4) | (C++17 起) |
void operator delete ( void* ptr, std::size_t sz ) noexcept; |
(5) | (C++14 起) |
void operator delete[]( void* ptr, std::size_t sz ) noexcept; |
(6) | (C++14 起) |
void operator delete ( void* ptr, std::size_t sz, std::align_val_t al ) noexcept; |
(7) | (C++17 起) |
void operator delete[]( void* ptr, std::size_t sz, std::align_val_t al ) noexcept; |
(8) | (C++17 起) |
可替换的定位解除分配函数 |
||
(9) | ||
void operator delete ( void* ptr, const std::nothrow_t& tag ) throw(); |
(C++11 前) | |
void operator delete ( void* ptr, const std::nothrow_t& tag ) noexcept; |
(C++11 起) | |
(10) | ||
void operator delete[]( void* ptr, const std::nothrow_t& tag ) throw(); |
(C++11 前) | |
void operator delete[]( void* ptr, const std::nothrow_t& tag ) noexcept; |
(C++11 起) | |
void operator delete ( void* ptr, std::align_val_t al, const std::nothrow_t& tag ) noexcept; |
(11) | (C++17 起) |
void operator delete[]( void* ptr, std::align_val_t al, const std::nothrow_t& tag ) noexcept; |
(12) | (C++17 起) |
非分配定位解除分配函数 |
||
(13) | ||
void operator delete ( void* ptr, void* place ) throw(); |
(C++11 前) | |
void operator delete ( void* ptr, void* place ) noexcept; |
(C++11 起) | |
(14) | ||
void operator delete[]( void* ptr, void* place ) throw(); |
(C++11 前) | |
void operator delete[]( void* ptr, void* place ) noexcept; |
(C++11 起) | |
用户定义的定位解除分配函数 |
||
void operator delete ( void* ptr, args... ); |
(15) | |
void operator delete[]( void* ptr, args... ); |
(16) | |
类特定的常规解除分配函数 |
||
void T::operator delete ( void* ptr ); |
(17) | |
void T::operator delete[]( void* ptr ); |
(18) | |
void T::operator delete ( void* ptr, std::align_val_t al ); |
(19) | (C++17 起) |
void T::operator delete[]( void* ptr, std::align_val_t al ); |
(20) | (C++17 起) |
void T::operator delete ( void* ptr, std::size_t sz ); |
(21) | |
void T::operator delete[]( void* ptr, std::size_t sz ); |
(22) | |
void T::operator delete ( void* ptr, std::size_t sz, std::align_val_t al ); |
(23) | (C++17 起) |
void T::operator delete[]( void* ptr, std::size_t sz, std::align_val_t al ); |
(24) | (C++17 起) |
类特定的定位解除分配函数 |
||
void T::operator delete ( void* ptr, args... ); |
(25) | |
void T::operator delete[]( void* ptr, args... ); |
(26) | |
类特定的常规销毁解除分配函数 |
||
void T::operator delete( T* ptr, std::destroying_delete_t ); |
(27) | (C++20 起) |
void T::operator delete( T* ptr, std::destroying_delete_t, std::align_val_t al ); |
(28) | (C++20 起) |
void T::operator delete( T* ptr, std::destroying_delete_t, std::size_t sz ); |
(29) | (C++20 起) |
void T::operator delete( T* ptr, std::destroying_delete_t, std::size_t sz, std::align_val_t al ); |
(30) | (C++20 起) |
解除分配先前由匹配的 operator new 或 operator new[] 分配的存储。这些解除分配函数由 delete 和 delete[] 表达式以及 定位 new 表达式 调用,以在销毁(或未能构造)具有动态存储期的对象后解除分配内存。它们也可以使用常规函数调用语法进行调用。
- 如果 ptr 不是空指针,并且满足以下条件之一,则行为未定义:
- 对于 operator delete,ptr 的值不表示先前调用 (可能已替换的) operator new(std::size_t) (对于重载 (1,5,9)) 或 operator new(std::size_t, std::align_val_t) (对于重载 (3,7,11)) 所分配的内存块的地址,并且该内存块尚未因对 operator delete 的中间调用而失效。
- 对于 operator delete[],ptr 的值不表示先前调用 (可能已替换的) operator new[](std::size_t) (对于重载 (2,6,10)) 或 operator new[](std::size_t, std::align_val_t) (对于重载 (4,8,12)) 所分配的内存块的地址,并且该内存块尚未因对 operator delete[] 的中间调用而失效。
重载 (1-8) 在每个翻译单元中隐式声明,即使未包含 <new> 头文件。
有关选择重载的条件,请参阅 delete 表达式。
目录 |
[编辑] 参数
ptr | - | 指向要解除分配的内存块的指针或空指针 |
sz | - | 传递给匹配的分配函数的大小 |
place | - | 在匹配的定位 new 中用作定位参数的指针 |
tag | - | 重载消歧标记,与非抛出 operator new 使用的标记匹配 |
al | - | 已分配对象或数组元素的对齐方式 |
args | - | 匹配定位分配函数的任意参数(可能包括 std::size_t 和 std::align_val_t) |
[编辑] 异常
所有解除分配函数都是 noexcept(true),除非在声明中另有规定。 |
(C++11 起) |
如果解除分配函数通过抛出异常终止,则行为未定义,即使它被声明为 noexcept(false)(C++11 起)。
[编辑] 全局替换
全局 operator
s new/delete 替换
#include <cstdio> #include <cstdlib> #include <new> // no inline, required by [replacement.functions]/3 void* operator new(std::size_t sz) { std::printf("1) new(size_t), size = %zu\n", sz); if (sz == 0) ++sz; // avoid std::malloc(0) which may return nullptr on success if (void *ptr = std::malloc(sz)) return ptr; throw std::bad_alloc{}; // required by [new.delete.single]/3 } // no inline, required by [replacement.functions]/3 void* operator new[](std::size_t sz) { std::printf("2) new[](size_t), size = %zu\n", sz); if (sz == 0) ++sz; // avoid std::malloc(0) which may return nullptr on success if (void *ptr = std::malloc(sz)) return ptr; throw std::bad_alloc{}; // required by [new.delete.single]/3 } void operator delete(void* ptr) noexcept { std::puts("3) delete(void*)"); std::free(ptr); } void operator delete(void* ptr, std::size_t size) noexcept { std::printf("4) delete(void*, size_t), size = %zu\n", size); std::free(ptr); } void operator delete[](void* ptr) noexcept { std::puts("5) delete[](void* ptr)"); std::free(ptr); } void operator delete[](void* ptr, std::size_t size) noexcept { std::printf("6) delete[](void*, size_t), size = %zu\n", size); std::free(ptr); } int main() { int* p1 = new int; delete p1; int* p2 = new int[10]; // guaranteed to call the replacement in C++11 delete[] p2; }
可能的输出
// Compiled with GCC-5 in C++17 mode to obtain the following: 1) op new(size_t), size = 4 4) op delete(void*, size_t), size = 4 2) op new[](size_t), size = 40 5) op delete[](void* ptr)
具有附加用户定义参数(“定位形式”,(15,16))的 operator delete 和 operator delete[] 的重载可以在全局作用域中声明,并由 new 表达式的匹配定位形式调用,如果正在分配的对象的构造函数抛出异常。
标准库的 operator delete 和 operator delete[] 的定位形式 (13,14) 不能被替换,并且只能在定位 new 表达式未使用 ::new 语法的情况下进行自定义,方法是提供具有匹配签名的类特定定位 delete (25,26):void T::operator delete(void*, void*) 或 void T::operator delete[](void*, void*)。
[编辑] 类特定重载
解除分配函数 (17-24) 可以定义为类的静态成员函数。如果提供,这些解除分配函数由 delete 表达式在删除此类对象 (17,19,21) 和数组 (18,20,22) 时调用,除非 delete 表达式使用了 ::delete 形式,该形式会绕过类作用域查找。static 关键字对于这些函数声明是可选的:无论是否使用该关键字,解除分配函数始终是静态成员函数。
delete 表达式从类作用域(数组形式在数组元素类的作用域中查找)开始查找适当的解除分配函数名称,如果没有找到成员,则像往常一样继续到全局作用域。请注意,根据名称查找规则,在类作用域中声明的任何解除分配函数都会隐藏所有全局解除分配函数。
如果正在删除的对象的静态类型与其动态类型不同(例如通过指向基类的指针删除多态对象时),并且静态类型中的析构函数是虚函数,则单个对象形式的 delete 会从其虚析构函数最终重写器的定义点开始查找解除分配函数的名称。无论运行时将执行哪个解除分配函数,operator delete 的静态可见版本必须可访问才能编译。在其他情况下,当通过指向基类的指针删除数组时,或通过指向具有非虚析构函数的基类的指针删除时,行为未定义。
如果未提供单参数重载 (17,18),但提供了接受 std::size_t 作为第二个参数的大小感知重载 (21,22),则会为正常解除分配调用大小感知形式,并且 C++ 运行时将要解除分配对象的大小作为第二个参数传递。如果两种形式都已定义,则调用不感知大小的版本。
#include <cstddef> #include <iostream> // sized class-specific deallocation functions struct X { static void operator delete(void* ptr, std::size_t sz) { std::cout << "custom delete for size " << sz << '\n'; ::operator delete(ptr); } static void operator delete[](void* ptr, std::size_t sz) { std::cout << "custom delete for size " << sz << '\n'; ::operator delete[](ptr); } }; int main() { X* p1 = new X; delete p1; X* p2 = new X[10]; delete[] p2; }
可能的输出
custom delete for size 1 custom delete for size 18
具有附加用户定义参数(“定位形式”,(25,26))的 operator delete 和 operator delete[] 的重载也可以定义为类成员。当失败的定位 new 表达式查找要调用的相应定位 delete 函数时,它会在检查全局作用域之前从类作用域开始查找,并查找与定位 new 签名匹配的函数。
#include <cstddef> #include <iostream> #include <stdexcept> struct X { X() { throw std::runtime_error("X(): std::runtime_error"); } // custom placement new static void* operator new(std::size_t sz, bool b) { std::cout << "custom placement new called, b = " << b << '\n'; return ::operator new(sz); } // custom placement delete static void operator delete(void* ptr, bool b) { std::cout << "custom placement delete called, b = " << b << '\n'; ::operator delete(ptr); } }; int main() { try { [[maybe_unused]] X* p1 = new (true) X; } catch (const std::exception& ex) { std::cout << ex.what() << '\n'; } }
输出
custom placement new called, b = 1 custom placement delete called, b = 1 X(): std::runtime_error
如果类级别的 operator delete 是模板函数,则它必须具有 void 返回类型,第一个参数为 void*,并且它必须有两个或更多参数。换句话说,只有定位形式可以是模板。无论其签名如何,模板实例都不是常规解除分配函数。模板 operator delete 的特化通过模板参数推导选择。
[编辑] 注意
在多态类上调用类特定的 T::operator delete 是静态成员函数通过动态调度调用的唯一情况。
以下函数必须是线程安全的
对这些函数中分配或解除分配特定存储单元的调用以单一总顺序发生,并且每次解除分配调用都先于此顺序中的下一次分配(如果有)。 |
(C++11 起) |
特性测试宏 | 值 | 标准 | 特性 |
---|---|---|---|
__cpp_sized_deallocation |
201309L |
(C++14) | 尺寸感知解除分配 |
__cpp_impl_destroying_delete |
201806L |
(C++20) | 销毁式 operator delete(编译器支持) |
__cpp_lib_destroying_delete |
201806L |
(C++20) | 销毁式 operator delete(库支持) |
[编辑] 缺陷报告
下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。
缺陷报告 | 应用于 | 发布时的行为 | 正确的行为 |
---|---|---|---|
CWG 220 | C++98 | 允许用户定义的解除分配函数抛出异常 | 从解除分配函数抛出异常 导致未定义行为 |
CWG 1438 | C++98 | 任何使用无效指针值的行为都未定义 | 只有间接引用和解除分配是 |
LWG 206 | C++98 | 替换 (2) 不影响 (10) 的默认行为 | 默认行为 相应改变 |
LWG 298 | C++98 | 替换 (1) 不影响 (9) 的默认行为 | 默认行为 相应改变 |
LWG 404 | C++98 | 可替换解除分配函数的替换 函数可以声明为 inline |
禁止,无需诊断 |
LWG 2458 | C++14 | 指定了接受 (void*, std::size_t, const std::nothrow_t&) 的重载,但从未被调用 |
移除了多余的重载 |
[编辑] 另请参阅
[static] (C++23) |
解除分配先前从 operator new 获取的内存( std::generator<Ref,V,Allocator>::promise_type 的公共静态成员函数) |
分配函数 (函数) | |
(在 C++17 中已弃用)(在 C++20 中已移除) |
释放未初始化存储 (函数模板) |
释放先前分配的内存 (函数) |