Namespaces
Variants

operator delete , operator delete[]

From cppreference.net
< cpp ‎ | memory ‎ | new
Utilities library
Memory management library
( exposition only* )
Allocators
Uninitialized memory algorithms
Constrained uninitialized memory algorithms
Memory resources
Uninitialized storage (until C++20)
( until C++20* )
( until C++20* )
( until C++20* )

Garbage collector support (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
(C++11) (until C++23)
定义于头文件 <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 )</

释放先前由匹配的 operator new operator new[] 分配的存储空间。这些释放函数通过 delete delete [ ] 表达式 以及 placement new 表达式 调用,用于在析构(或构造失败)具有动态存储期的对象后释放内存。也可以通过常规函数调用语法来调用这些函数。

1-12) 可替换的 解分配函数。标准库为这些函数提供了默认实现,关于默认实现的效果,请参见 下文
1-8) delete delete [ ] 表达式调用。使任何非空的 ptr 失效。
9-12) 由布置 new 表达式在 初始化失败 时调用。 operator delete [ ] 会使任何非空的 ptr 失效。
如果 ptr 不是空指针且满足以下任一条件,则行为未定义:
13,14) 当表达式中的任何初始化部分因抛出异常而终止时,由调用了 非分配布置分配函数 的布置 new 表达式调用。不执行任何操作。
15-30) delete delete [ ] 以及布置 new 表达式调用的用户自定义释放函数。
27-30) 若已定义, delete 表达式在调用 operator delete 之前不会执行 * ptr 的析构函数。相反,直接调用析构函数(例如通过 ptr - > ~T ( ) ; )成为此 operator delete 的职责。

重载版本 ( 1-8 ) 在每个翻译单元中隐式声明,即使未包含 <new> 头文件。

请参阅 delete 表达式 了解选择重载的标准。

目录

参数

ptr - 指向要释放的内存块的指针或空指针
sz - 传递给匹配分配函数的大小参数
place - 在匹配的 placement new 中用作布局参数的指针
tag - 与非抛出 operator new 所用标签匹配的重载消歧标签
al - 已分配对象或数组元素的对齐要求
args - 与布局分配函数匹配的任意参数(可能包含 std::size_t std::align_val_t

异常

所有解分配函数均为 noexcept ( true ) ,除非在声明中另有说明。

(since C++11)

如果解分配函数因抛出异常而终止,则行为未定义 ,即使它被声明为 noexcept ( false ) (C++11 起)

全局替换

重载版本 ( 1-12 ) 可替换的 。默认版本的效果如下:

1) ptr 为空指针,则不执行任何操作。否则,回收先前通过 operator new 调用所分配的存储空间。
2) 调用 operator delete ( ptr ) ,如同重载版本 (1) 能够回收先前通过 operator new [ ] 调用所分配的存储空间。
3) (1)
4) 调用 operator delete ( ptr, al ) ,如同重载版本 (3) 能够回收先前通过 operator new [ ] 调用所分配的存储空间。
5) 调用 operator delete ( ptr )
6) 调用 operator delete [ ] ( ptr )
7) 调用 operator delete ( ptr, al )
8) 调用 operator delete [ ] ( ptr, al )
9) 调用 operator delete ( ptr )
10) 调用 operator delete [ ] ( ptr )
11) 调用 operator delete ( ptr, al )
12) 调用 operator delete [ ] ( ptr, al )

全局 operator 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)

具有额外用户定义参数的 operator delete operator delete [ ] 重载版本(“placement 形式”, ( 15,16 ) )可以像常规函数一样在全局作用域中声明,当正在分配对象的构造函数抛出异常时,会被匹配的 new 表达式的 placement 形式调用。

标准库中 operator delete operator delete [ ] 的布置形式 ( 13,14 ) 不可被替换,且仅当布置 new 表达式未使用 :: new 语法时,才能通过提供具有匹配签名的类特定布置删除函数 ( 25,26 ) 进行自定义: void T :: operator delete ( void * , void * ) void T :: operator delete [ ] ( void * , void * )

类特定重载

释放函数 ( 17-24 ) 可定义为类的静态成员函数。当删除此类对象 ( 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>
// 特定大小的类专属释放函数
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

具有额外用户定义参数的 operator delete operator delete [ ] 重载版本("placement 形式", ( 25,26 ) )也可以定义为类的成员函数。当失败的 placement new 表达式寻找对应的 placement delete 函数进行调用时,会在检查全局作用域之前先从类作用域开始查找,并寻找与 placement new 签名匹配的函数:

#include <cstddef>
#include <iostream>
#include <stdexcept>
struct X
{
    X() { throw std::runtime_error("X(): std::runtime_error"); }
    // 自定义 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);
    }
    // 自定义 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 * ,且必须具有两个或更多参数。换言之,只有 placement 形式的 operator delete 才能作为模板。无论其签名如何,模板实例永远不会成为常规的释放函数。模板 operator delete 的特化版本通过 模板实参推导 进行选择。

注释

对多态类调用特定类的 T :: operator delete 是通过动态分派调用静态成员函数的唯一场景。

以下函数要求是线程安全的:

对这些函数分配或释放特定存储单元的调用以单一全序发生,且每个这样的释放调用 先发生于 该顺序中的下一个分配调用(如果存在)。

(since C++11)
功能测试 标准 功能特性
__cpp_sized_deallocation 201309L (C++14) 带尺寸的释放操作
__cpp_impl_destroying_delete 201806L (C++20) 销毁式 delete 运算符(编译器支持)
__cpp_lib_destroying_delete 201806L (C++20) 销毁式 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 中移除)
释放未初始化的存储空间
(函数模板)
释放先前分配的内存
(函数)