本文目录

专业名称

new operator:new操作符,new表达式

operator new:new运算符

placement new:定位new

delete operator:delete操作符,delete表达式

operator delete:delete运算符

delete没有placement版本 各个版本的解释 1. new operator

new operator:new表达式

实际上只是一个称呼,是三个操作的集合。

  • 申请内存(调用operator new)
  • 在内存上调用构造函数
  • 将申请的内存空间返回

delete operator:delete表达式

delete operator也是一个称呼,是两个操作的集合。

  • 调用析构函数
  • 调用operator delete释放空间

operator new/delete:new/delete运算符

  • 分为::operator new与Class::operator new。一个是全局的,一个是类内部定义的。平时听到的new与delete重载,就是重载这里的new/delete。注意点如下。

    -- class中的new与delete是隐式静态的。也就是不管你有没有声明为static,它都是static的。

    -- 非static成员在没有实例存在的情况下是不能使用的。而当我们定义一个实例时,很明显此时实例还没有存在,这个时候不可能调用一个非static的成员,所以operator new/delete必须是static的。

    -- static成员是不会隐式传入this指针的,所以在operator new/delete中不能使用任何类的非静态成员。

    -- 非静态的成员函数在编译器会在argument list的第一个位置插入一个本身类型的this指针。所以一个函数名,参数列表与全局函数一模一样的函数,也不会出现二义性问题。但是一个static成员函数,是不会隐私插入this指针的,所以当我们在class中重载new/delete时,客户定义实例时需要确认是否在全局也定义了new,否则会产生二义性。举个栗子。
//VS2013
class StringClass
{
public:
void *operator new(size_t size);
};
void *operator new(size_t size);
void main()
{
//1. 调用operator new申请内存
//2. 调用构造函数
//二义性,同时存在两个可用的operator new
StringClass *p = new StringClass;
}

-- PS:关于函数匹配规则,一开始接触觉得很简单,当某天触发了bug的时候就会“书到用时方恨少”,这方面推荐两个连接给大家。看这里还有这里

  • 关于operator new/delete标准库提供了8个版本,这些版本的new/delete我们都是可以重载的。如下。
//八个版本中有2个new版本有可能抛出异常
void* operator new(size_t t);
void* operator new[](size_t t);
void* operator delete(void*)noexcept;
void* operator delete[](void*)noexcept; //nothrow_t为空结构体,用于区分上面与下面运算符
void* operator new(size_t t,nothrow_t&)noexcept;
void* operator new(size_t t,nothrow_t&)noexcept;
void* operator delete(void *,nothrow_t&)noexcept;
void* operator delete[](void *,nothrow_t&)noexcept;

理论这么多,下面先来个栗子。

//StringClass.h
//自己定义的一个类
#pragma once
#include<iostream>
using namespace std;
class StringClass
{
public:
StringClass(){
cout << "string class construtor" << endl;
}
~StringClass(){
cout << "string class destrutor" << endl;
}
void *operator new(size_t size){
cout << "Class::operator new(size_t) " << size << endl;
void *ret = (void*)malloc(size);
return ret;
}
static void operator delete(void *p){
free(p);
cout << "Class::operator delete(void*)" << endl;
}
};
void main(){
StringClass *p = new StringClass;
delete p;
}

结果如下。在new表达式中,先调用了operator new,再调用constructor,最后将指针返回给p。最后在释放的时候,先调用了destructor,再调用了operator delete。

既然使用到了delete,怀着“授人以鱼不如授人以渔”的想法,推荐读者了解一下crtdbg这个头文件,谷歌一下就知道了。

placement new:定位new

  • 在堆里面申请内存,需要寻找空的内存,多次申请释放还会产生大量的内存碎片,但这个神器会帮你解决这些问题。使用它,你只需要申请一次内存,就能够多次创建不同的实例,具体参见下例的讲解。
  • 前面提到,new operator会有三个步骤,第一是申请内存,第二是调用构造函数,第三是返回指针。定位new,之所以为定位,是因为我们事先分配好了内存,并且指定了必须使用这块内存进行初始化。共有四种语法如下。
new (place_address) type;
new (place_address) type (initializers);
new (place_address) type[size];
new (place_address) type[size]{braced initializers list};

对第一种我们举个栗子。

#pragma once
#include<iostream>
using namespace std; class StringClass
{
public:
StringClass(){
cout << "string class construtor" << endl;
}
~StringClass(){
cout << "string class destrutor" << endl;
}
//比起前面的代码,增添了这个函数
static void *operator new(size_t size, void *p) {
cout << "Class::operator new(size_t,void*)" << endl;
return p;
}
}; void main()
{
StringClass *buf = (StringClass*)malloc(sizeof(StringClass));
memset(buf, 0, sizeof(StringClass));
StringClass *p2 = new(buf) StringClass();
p2->~StringClass();
free(buf);
}
  • 上面总共有五个步骤:

    -- 上述代码先使用申请了一块内存,大小用户根据自己需要去决定。

    -- 初始化这块内存。

    -- 然后在上面调用了StringClass的构造函数,再令p2指向他。

    -- 然后p2使用完毕在上面调用自己的析构函数。

    -- 程序结束前将这块内存释放。假如有需要创建另外一个对象,我们可以不用申请新的内存,继续使用这块内存。

  • 与operator new不同的是,使用placement new是无需定义operator delete的,如上,整个过程压根就没有用到它。相反,假如你在客户端使用了它,还会导致调用析构函数的多次调用。

警告

除了placement new外,其他版本的new与delete都要成对地重载。因为你重载了编译器的new,它不知道delete需要为你做些什么。

总结

  • new总共分成3类,new operator是一个操作的集合,不是具体的运算符。可以重载的叫做operator new。
  • 使用了placement new,必须定义void *operator new(siez_t,void *p),不用定义operator delete。而除了placement new,若定义了operator new/delete需要成对定义。
  • 所有的operator new都是隐式静态的,为了让代码易于阅读,你应该加上static
  • 所有的operator new都应该返回void*

最后的疑问

在很多书籍以及博客中,都指出了下面这个operator new不能被重载。在C++Primer第五版,白色封面的那本,P727最下方中,明确指出了它不能重载,但实际上,使用placement new的时候我们却不得不重载这个版本。关于这一点,希望有高人能够给出意见

C++基础——new与delete的更多相关文章

  1. Python3 tkinter基础 Entry insert delete 点击按钮 向输入框赋值 或 清空

             Python : 3.7.0          OS : Ubuntu 18.04.1 LTS         IDE : PyCharm 2018.2.4       Conda ...

  2. C++基础 new和delete

    1.new delete 的使用 (1)基本数据类型 ); delete p; int *p = (int *)malloc(sizeof(int)); *p = ;free(p); (2)数组 ]; ...

  3. 关于delete和delete[]

    [精彩] 求问delete和delete[] 的区别??http://www.chinaunix.net/jh/23/311058.html C++告诉我们在回收用 new 分配的单个对象的内存空间的 ...

  4. 【转】清理Kylin的中间存储数据(HDFS & HBase Tables)

    http://blog.csdn.net/jiangshouzhuang/article/details/51290399 Kylin在创建cube过程中会在HDFS上生成中间数据.另外,当我们对cu ...

  5. Memcached详解

    Memcached介绍 Memcached是什么? Free & open source, high-performance, distributed memory object cachin ...

  6. MySQL auto_increment_increment 和 auto_increment_offset

    参考这一篇文章:(不过我对这一篇文章有异议) http://blog.csdn.net/leshami/article/details/39779509 1:搭建测试环境 create table t ...

  7. SQL Server中一些有用的日期sql语句

    SQL Server中一些有用的日期sql语句 1.一个月第一天的 SELECT DATEADD(mm, DATEDIFF(mm,0,getdate()), 0) 2.本周的星期一 SELECT DA ...

  8. 02 DML(DataManipulationLanguage)

    1.插入记录     基本语法 :         INSERT INTO tbl_name (col_name ,col_name1,..,col_nameN) VALUES (val1,val2, ...

  9. 转载:STL四种智能指针

    转载至:https://blog.csdn.net/K346K346/article/details/81478223 STL一共给我们提供了四种智能指针: auto_ptr.unique_ptr.s ...

随机推荐

  1. 51nod 1202 子序列个数

    1202 子序列个数  题目来源: 福州大学 OJ 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题  收藏  关注 子序列的定义:对于一个序列a=a[1],a[2] ...

  2. [异常记录-12]Web Deploy部署:未能连接到远程计算机,请确保在远程计算机上安装了 Web Deploy 并启动了所需的进程("Web Management Service")

    Web Deploy 安装 请参考:图文详解远程部署ASP.NET MVC 5项目 如此安装后还不行,  可以在卸载后重新安装 Web Deploy 时,不要选那个经典还是典型的安装选项,选自定义安装 ...

  3. 3:JavaScript中的各类语句

    上面我们说完了数据类型和基本运算  下面说的就是各种类型的语句 --- ---------------------------------------------------------------- ...

  4. hdu 6020 MG loves apple 恶心模拟

    题目链接:点击传送 MG loves apple Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 262144/262144 K (Ja ...

  5. Cocos2d-x学习笔记(十二)3D特效

    特效类即是GridAction类,其实就是基于网格的3D动作类.需开启OpenGL的深度缓冲,否则容易3D失真. 下边是一个snippet,创建网格对象,并将其添加到当前layer:同时,将进行3D特 ...

  6. BN层

    论文名字:Batch Normalization: Accelerating Deep Network Training by  Reducing Internal Covariate Shift 论 ...

  7. C++指针总结

    在C++中通过动态创建的对象,我们只能获得一个指针,并通过指针控制它.指针是存放对象的内存地址值,更准确的描述是对象的起始地址值.每一个指针都有一个相关的类型,不同数据类型的指针之间的区别不在指针的描 ...

  8. Javascript 面向对象-继承

    JavaScript虽然不是面向对象的语言,但是我们通过构造可以让其支持面向对象,从而实现继承.重写等面向对象的特性.具体代码如下: //创建类Person function Person(age,n ...

  9. B2B、B2C、C2C、O2O 和 P2P 的含义

    B2C(Business-to-Customer)商家对客户 我开一家公司卖东西,你来买,即B2C.生活中常用的比如我们经常在天猫旗舰店上面购物,天猫入驻的都是商家,而我们买东西的就是客户,这就是B2 ...

  10. [转]TOMCAT启动提示NB: JAVA_HOME should point to a JDK not a JRE解决

    来源:http://blog.csdn.net/caozhongyan/article/details/6602759 本人使用的Tomcat版本为apache-tomcat-6.0.18(用的是解压 ...