重载new和delete

new的操作:

(1)调用operator new函数获取内存空间

(2)在分配的内存空间中调用构造函数构造对象

delete操作:

(1)调用析构函数销毁对象

(2)调用operator delete函数释放对象所占的空间。

其实operator new和operator delete的底层实现是调用了malloc和free。这两个函数定义在<cstdlib>头文件中。

malloc函数接受一个表示待分配字节数的size_t,返回指向分配空间的指针或者返回0以表示分配失败。

free函数接受一个void*,它是malloc返回的指针的副本。free将相关内存返回给系统。调用free(0)没有任何意义。

以下是简单版本的malloc和free的实现

void* operator new (size_t size)
{
    if(void* mem = malloc(size))
        return mem;
    else
        throw bad_alloc();
}

void operator delete(void* mem) noexcept
{
    free(mem);
}

 

如果应用程序希望控制内存分配的过程,则他们需要定义自己的operator new函数和operator delete函数。

当编译器发现一条new表达式或delete表达式后,将在程序中查找可供调用的operator函数。如果被分配(释放)的对象时类类型,则编译器首先在类及其基类的作用域中查找。如果此时该类中含有operator new成员或operator delete成员,则相应的表达式将调用这些成员。否则,编译器在全局作用域中查找匹配的函数。此时如果编译器找到了用户自定义的版本,则使用该版本执行new表达式或delete表达式,如果没有找到,则使用标准库定义的版本。

 

operator new接口和operator delete接口:

一共有8个重载版本,前面四个可能抛出bad_alloc异常,后面四个不会抛出异常。

//这些版本可能抛出异常
void* operator new(size_t);    //分配一个对象
void* operator new[](size_t);    //分配一个数组
void* operator delete(void*) noexcept;    //释放一个对象
void* operator delete[](void*) noexcept;    //释放一个数组

//这些版本承诺不会抛出异常
void* operator new(size_t, nothrow_t&) noexcept;    
void* operator new[](size_t, nothrow_t&) noexcept;    //分配一个数组
void* operator delete(void*, nothrow_t&) noexcept;    //释放一个对象
void* operator delete[](void*, nothrow_t&) noexcept;    //释放一个数组

应用程序可以自定义上面函数版本中的任何一个,前提是自定义的版本必须位于全局作用域或者类作用域中。

我们将上述运算符函数定义为成员类的成员时,他们是饮食静态的。因为operator new用在构造对象之前,而operator delete用在对象销毁之后。所以这两个成员必须是静态的。而且它们不能操纵类的任何数据成员。

 

对于operator new和operator new[]函数来说,它们的返回类型必须是void*。第一个形参的类型必须是size_t且该形参不能含有默认实参。如果我们想要自定义operator new函数,则可以为它提供额外的形参。不过此时,用到这些自定义函数的new表达式必须使用new的定位形式。

 

有一种operator new无论如可也不能被用户重载

void *operator new(size_t, void*);    //不允许重新定义这个版本

这种形式只供标准库使用,不能被用户重新定义。

 

对于operator delete和operator delete[]函数来说,他们的返回类型必须是void,第一个形参的类型必须是void*。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在C++中,可以重载newdelete运算符以定制动态内存管理的行为。重载new运算符可以用于自定义内存分配的方式,而重载delete运算符可以用于自定义内存释放的方式。 重载new运算符的一种常见方式是定义一个全局的new运算符函数,并使用该函数来执行内存分配。例如: ```cpp void* operator new(size_t size) { // 自定义内存分配逻辑 void* ptr = malloc(size); // 检查分配是否成功 if (ptr == nullptr) { throw std::bad_alloc(); } return ptr; } ``` 重载delete运算符的一种常见方式是定义一个全局的delete运算符函数,并使用该函数来执行内存释放。例如: ```cpp void operator delete(void* ptr) noexcept { // 自定义内存释放逻辑 free(ptr); } ``` 需要注意的是,如果重载new运算符,通常也需要相应地重载delete运算符,以确保内存的正确释放。 可以根据需要重载其他版本的newdelete运算符,例如带有额外参数的newdelete运算符,数组形式的newdelete运算符等。重载这些运算符时需要遵循一定的规则和约定,确保正确性和可靠性。 值得注意的是,C++11引入了更加灵活和安全的内存管理方式,例如智能指针(如std::shared_ptr和std::unique_ptr)和RAII(资源获取即初始化)等,这些方式可以减少手动管理内存的复杂性和错误。因此,在使用newdelete运算符进行内存管理之前,建议先考虑这些更高级的内存管理工具。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值