C++的new&delete
C++的new&delete
new的过程
new的过程:先分配memory,再调用ctor
我们常用的创建对象的方法有两种
Complex c(1,2); //栈
Complex *pc = new Complex(1,2); //堆
第一种创建出来的对象将保存在栈上,第二种则在堆上,必须手动回收内存空间(通过delete)
为了解释new的过程,我们先建立一个Complex类
class Complex
{
public:
Complex(...) {...}//构造函数
...
private:
double real;
double imag;
};
当我们使用new构建Complex类对象的时候
Complex *pc = new Complex(1,2);
当我们使用new这一个动作,在堆上动态创建一个对象时,编译器实际上帮你做了三件事:
Complex *pc;
//1.分配内存(调用 operator new() 函数)
void* memory = operator new(sizeof(Complex));
//2.转型
pc = static_cast<Complex*>(memory);
//3.调用构造函数
pc->Complex::Complex(1,2);
- 分配内存:
operator new也是一个函数,其内部调用malloc(n),拿到sizeof(Complex)大小的内存空间;这时候我们得到指向内存空间始址的指针memory,它是一个指向viod类型的指针 - 转型:用
static_cast函数,把步骤①得到的指针memory(这是一个pointer to void)转换为pointer to Complex,并将其赋值到pc(步骤①和②可以写在一起) - 调用构造函数:步骤②得到的指针pc指向的内存空间,即为新对象的起始内存地址;于是编译器将通过指针pc调用对象的构造函数
所以从结果上看,这两段代码是等效的
//代码1.
Complex *pc = new Complex(1,2);
//代码2.
Complex *pc;
void* memory = operator new(sizeof(Complex));
pc = static_cast<Complex*>(memory);
pc->Complex::Complex(1,2);
malloc和new的区别在于,当malloc失败时,它不会调用分配内存失败处理程序new_handler,因此我们还是要尽可能的使用new,除非有一些特殊的需求
delete的过程
delete的过程:先调用dtor,再释放memory
我们再建立一个包含指针的类String:
class String {
public:
...
~String()
{delete[] m_data;}
...
private:
char* m_data;
};
当我们试用new&delete时:
String* ps = new String("HELLO");
...
delete ps;
编译器在delete这里实际上帮你做了两件事:
String::~String(ps); //1.调用析构函数
operator delete(ps); //2.释放内存
- 调用析构函数:由于String类是包含指针的,所以设计时不能使用默认析构函数,而是重载一个符合需求的析构函数,在我们delete ps时,编译器第一步就是调用我们重载后的析构函数(没有重载则调用默认)
- 释放内存:
operator delete和operator new一样也是一个函数,其内部调用free(ps)
new的三种形态
有的朋友可能被上面的new和operator new搞晕了,实际上在C++中提到new,至少可能代表以下三种含义:new operator,operator new,placement new
new operator
new operator 就是 new 操作符,不能被重载
我们上面所说的new,都是指new operator,也就是我们平时使用的new
operator new
operator new 是函数,可以被重载,new operator 调用它用来分配内存,通过重载它,可以改变 new operator 的功能
默认有三种
void* operator new (std::size_t size) throw (std::bad_alloc);
void* operator new (std::size_t size, const std::nothrow_t& nothrow_constant) throw();
void* operator new (std::size_t size, void* ptr) throw();
- 第一种分配size个字节的存储空间,并将对象类型进行内存对齐。如果成功,返回一个非空的指针指向首地址。失败抛出bad_alloc异常。 (
A* a = new A;调用第一种 - 第二种在分配失败时不抛出异常,它返回一个NULL指针。 (
A* a = new(std::nothrow) A;//调用第二种 - 第三种是 placement new 版本,它本质上是对 operator new 的重载,定义于#include 中,它不分配内存,调用合适的构造函数在 ptr 所指的地方构造一个对象,之后返回实参指针ptr,下文细谈
重载 operator new
class Complex
{
public:
Complex(...) {...}//构造函数
...
//重载第一种
void* operator new(size_t size){
cout << "operator new called\n" << endl;
//通过::operator new调用了原有的全局的new
return ::operator new(size);
}
void operator delete(void* pointer)
{
cout << "operator delete" << endl;
::operator delete(pointer);
}
private:
double real;
double imag;
};
然后你可以直接调用 Complex::operator new(),或者使用 new 来调用
int main()
{
Complex* pc2 = new Complex(1,2);
}
输出
operator new called
operator delete
这里通过::operator new调用了原有的全局的new,也就相当于是在分配内存之前输出一句话
如上代码所示,delete 也有 delete operator 和 operator delete 之分,后者也是可以重载的。并且,如果重载了 operator new,就应该也相应的重载 operator delete,这是良好的编程习惯
重载 operator new 需要注意以下几点:
- 重载时,返回类型必须声明为void*
- 重载时,第一个参数类型必须为表达要求分配空间的大小(字节),类型为 size_t
- 重载时,可以带其它参数
带其他参数的重载:
void* operator new(size_t size, string str) {
cout << "operator new called\n" << endl;
cout << "with string:" << str << endl;
//通过::operator new调用了原有的全局的new
return ::operator new(size);
}
调用时就可以这样操作
Complex* pc = new("Test") Complex(1,2);
placement new
placement new 是 c++ 实现的 operator new 版本,它的实现如下
// Default placement versions of operator new.
inline void* operator new(std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT
{ return __p; }
inline void* operator new[](std::size_t, void* __p) _GLIBCXX_USE_NOEXCEPT
{ return __p; }
// Default placement versions of operator delete.
inline void operator delete (void*, void*) _GLIBCXX_USE_NOEXCEPT { }
inline void operator delete[](void*, void*) _GLIBCXX_USE_NOEXCEPT { }
//@}
可以看到实际上它就返回了传进来的地址,根据operator的第二个例子,通过重载全局的operator new之后,new函数的操作就被改变了。也就能猜出,在调用new的时候参数需要加上一个地址,placement new的功能就是在这个地址之上进行构造。
placement new 的使用步骤如下:
//1.分配内存
char* buff = new char[ sizeof(Complex) * N ];
memset( buff, 0, sizeof(Foo)*N );
//2.构建对象
Complex* pc = new (buff)Complex;
//3.使用对象
pc->XXXXXX();
//4.析构对象,显式的调用类的析构函数
pc->~Complex();
//5.销毁内存
delete[] buff;
上面5个步骤是标准的placement new的使用方法
placement new 是用来实现定位构造的,因此可以实现 new operator 三步操作中的调用构造函数这一步(在取得了足够内存空间后,在这块内存空间是上构造一个对象)
上面写的pc->Complex::Complex(1,2);这句话并不是一个标准的写法,正确的写法是使用 placement new:
#include <new.h>
int main()
{
char* buff = new char[ sizeof(Complex) ];
Complex* pc = new(buff) Complex(1,2);
...
}
placement new 它实现了在指定内存地址上用指定类型的构造函数来构造一个对象的功能,这块指定的地址既可以是栈,又可以是堆,placement 对此不加区分
除非特别必要,不要直接使用placement new ,这毕竟不是用来构造对象的正式写法,只不过是new operator的一个步骤而已。使用new operator地编译器会自动生成对placement new的调用的代码,因此也会相应的生成使用delete时调用析构函数的代码
如果是像上面那样在栈上使用了placement new,则必须手工调用析构函数,这也是显式调用析构函数的唯一情况
pc->~Complex();
当我们觉得默认的new operator对内存的管理不能满足我们的需要,而希望自己手工的管理内存时,placement new就有用了。STL中的allocator就使用了这种方式,借助placement new来实现更灵活有效的内存管理。
C++的new&delete的更多相关文章
- 如何区别数据库删除语句drop与delete与truncate?
1.delete:删除数据表中的行(可以删除某一行,也可以在不删除数据表的情况下删除所有行) 删除某一行:delete from 数据表名称 where 列名称=值: 删除所有行:delete*fro ...
- 数据库设计中的Soft Delete模式
最近几天有点忙,所以我们今天来一篇短的,简单地介绍一下数据库设计中的一种模式——Soft Delete. 可以说,该模式毁誉参半,甚至有非常多的人认为该模式是一个Anti-Pattern.因此在本篇文 ...
- 关于JavaScript中的delete操作
关于JavaScript中的delete操作 看到一道题,是这样的: (function(x){ delete x; return x; })(1); 1 null undefined Error 我 ...
- Git异常:Cannot delete the branch 'test1' which you are currently on
GitHub实战系列汇总:http://www.cnblogs.com/dunitian/p/5038719.html ———————————————————————————————————————— ...
- HTTP Method详细解读(`GET` `HEAD` `POST` `OPTIONS` `PUT` `DELETE` `TRACE` `CONNECT`)
前言 HTTP Method的历史: HTTP 0.9 这个版本只有GET方法 HTTP 1.0 这个版本有GET HEAD POST这三个方法 HTTP 1.1 这个版本是当前版本,包含GET HE ...
- IIS7.5上的REST服务的Put,Delete操作发生HTTP Error 405.0 - Method Not Allowed 解决方法
WebDAV 是超文本传输协议 (HTTP) 的一组扩展,为 Internet 上计算机之间的编辑和文件管理提供了标准.利用这个协议用户可以通过Web进行远程的基本文件操作,如拷贝.移动.删除等.在I ...
- ASP.NET Core 中文文档 第二章 指南(4.10)检查自动生成的Detail方法和Delete方法
原文 Examining the Details and Delete methods 作者 Rick Anderson 翻译 谢炀(Kiler) 校对 许登洋(Seay).姚阿勇(Mr.Yao) 打 ...
- new/delete重载
在c++中,有时我们需要在运行阶段为一个变量分配未命名的内存,并使用指针来访问它,这里就可以用到new关键字.另外需要指出的是,new分配的内存块通常与常规变量分配的内存块不同,常规变量的值都储存在被 ...
- EC笔记:第三部分:16成对使用new和delete
我们都知道,申请的资源,使用完毕后要释放.但是这个释放动作,一定要注意. 举个例子,很多人动态分配的资源,在使用之后,往往直接调用了delete,而不管申请资源的时候用的是new还是new[]. 如下 ...
- Spring boot: Request method 'DELETE' not supported, Request method 'PUT' not supported, Request method 'POST' not supported
GET,POST,PUT,DELETE, Spring都支持,不要怀疑Spring, 一定是前端发送的rest 请求和后端的响应不匹配, 查找原因以及解决办法, 很简单 用chrome打开F12控制台 ...
随机推荐
- HTML列表标签
<ul>无序列表 有2个属性 1.compact 属性: 规定列表呈现的效果比正常情况更小巧.没啥作用 2.type 属性 disc小圆点 square小方块 circle小圆圈(默认) ...
- 链接测试工具:Xenu
Xenu 是一款深受业界好评,并被广泛使用的死链接检测工具.时常检测网站并排除死链接,对网站的 SEO(搜索引擎优化) 非常重要,因为大量死链接存在会降低用户和搜索引擎对网站的信任. 最大支持100线 ...
- wireshark抓pc上的包
简介:wirkshark是全世界最广泛的网络封包分析软件之一. 软件用途: 网络管理员:我用它检测网络问题, 网络安全工程师:我用它检查资讯安全相关问题, 开发者:我用它为新的通讯协定除错, 普通使用 ...
- 10day 字符集优化 重点
什么是字符编码: 可以实现对非英文字符的支持 计算机操作系统是谁发明的? --美国人--英语 支持中文比较好的编码: UTF-8 比较通用 GBK 比较国际通用 出现乱码的原因: 01. 系统字符集设 ...
- requests.packages.urllib3.exceptions.ProxySchemeUnknown: Not supported proxy scheme
python3 -m pip install -U requests[socks]
- JFrog推出全球首个支持混合云架构,端到端的通用DevOps平台 ——JFrog Platform
JFrog Platform,基于屡获殊荣的JFrog Artifactory制品仓库的独特能力,通过多合一的体验提供DevSecOps.CI / CD和软件分发的解决方案. 2020 ...
- 【14】 DFS 机器人活动范围 (static插曲)
题目 地上有一个m行n列的方格,从坐标 [0,0] 到坐标 [m-1,n-1] .一个机器人从坐标 [0, 0] 的格子开始移动,它每次可以向左.右.上.下移动一格(不能移动到方格外),也不能进入行坐 ...
- 安装apache ActiveMQ
笔者环境 操作系统centos 6.9 java jdk 1.8 activemq版本 5.15.9 a)安装activemq需要jdk 环境,这里使用的是 jdk 1.8 yum安装jdk比较简单, ...
- Func<T,TResult>代理
.NET平台已经发生了很多变化,最近决定好好的系统的学习一下了,开发做了这么多年,老实说很多时候都是在吃老本,这样下去不行的... 今天学习的是Func<T,TResult>,它是新的委托 ...
- Python出现Could not find a version that satisfies the requirement openpyxl (from versions: )
一.环境使用python3.7时,用pip安装openpyxl出现如下错误: 系统环境:windows10家庭版Python版本:python3.7.1IDE:sublime_text 3二. 解决方 ...