理解各种不同含义的 new 和 delete
new operator new操作符
operator new 操作符new
placement new 定位new
string *ps = new string("Memory Management");
这里的new是new 操作符。这里共有两个步骤的工作要做:
①为对象申请空间 ②调用构造函数初始化内存中的对象
new 操作符总是做这两件事,不可以任何方式改变其行为。
但是你可以改变步骤①如何为对象申请空间。
new 操作符是通过 operator new这个函数为对象申请空间。
operator new 的声明形式
void* operator new(size_t size);//在做其他形式重载时也要保证第一个参数必须为size_t类型
函数的返回值是 void*,参数size确定分配多少内存
该函数调用时与普通函数一样
ST *pt = operator new(sizeof(ST));
placement new的作用是在已经被分配但是尚未处理的(raw)内存中构造一个对象。它是一个特殊的operator new
void* operator new(size_t, void *p)//参数size_t没有名字,但是为了防止编译器警告必须加上
{
return p;
}
在使用placement new时调用者已经获得了指向内存的指针,因为调用者知道对象应该放在
哪里。placement new必须做的就是返回传递给它的指针。
总结:如果你想在堆上建立一个对象,应该使用new 操作符,它既分配内存又为对象调用构造函数;
如果你只想分配内存,就调用operator new函数,它不会调用构造函数;
如果你想定制自己的在堆对象被建立时的内存分配过程,你应该写自己的operator new 函数,然后使用
new操作符,new操作符会调用你定制的operator new。
如果你想在一块已经获得指针的内存里建立一个对象,应该使用palcement new.
#include<iostream>
#include<vld.h>
using namespace std;
class ST
{
private:
char *ptr;
public:
ST(const char *str = ""):ptr("")
{
cout<<"Object was built. "<<endl;
if(str == NULL)
{
this->ptr = new char[1];
this->ptr[0] = '\0';
}
else
{
this->ptr = new char [strlen(str)+1];
strcpy(ptr, str);
}
}
~ST()
{
cout<<"Object was free. "<<endl;
delete []this->ptr;
ptr = NULL;
}
void* operator new(size_t, void *p)//placement new
{
return p;
}
void operator delete(void *p)
{
cout<<"---"<<endl;
free(p);
}
};
void* operator new(size_t t)
{
void *p = malloc(t);
return p;
}void* operator new[](size_t t)
{
void *p = malloc(t);
return p;
}
void operator delete[](void *p)
{
cout<<"--[][]\n";
free(p);
return;
}
void main()
{
ST *pt = (ST*)operator new (sizeof(ST)); //使用重载的operator new 为对象pt分配空间
new(pt)ST("Hello"); //这里调用placement new,同时也调用了构造函数。前两行代码等价于在未重载new时 ST *pt = new ST("Hello");
pt->~ST(); //通过pt指针可直接调用析构函数,但是不可调用构造函数
operator delete (pt); //调用operator delete函数 后行代码等价于在未重载delete时 delete pt
运行结果
为了避免内存泄漏,每个动态内存分配必须与一个等同相反的deallocation对应。
函数operator delete与delete操作符的关系和operator new与new操作符一样。
注意,在使用placement new在内存建立对象时,应避免使用delete操作符。
应该显式调用对象的析构函数来解除构造函数的影响。
在为对象数组申请空间时必可重载 operator new[] 函数而不再使用operator new函数。
例如
ST *pt = new ST[3];
这个过程共调用3次构造函数
将上例主函数修改为
void main()
{
ST *pr = new ST[3];
delete[] pr;//[]一定不可缺少
}
运行结果为
理解各种不同含义的 new 和 delete的更多相关文章
- 阅读《Effective C++》系列
<Effective C++>条款07:为多态基类声明virtual析构函数 这样做主要是为了防止内存泄漏,见我hexo博客. C++的虚析构函数 <Effective C++> ...
- More Effective C++ (1)
简单分析总结了more effective c++ 的前十个条款: 剩下的条款to be continue~ 1.仔细区分指针和引用引用必须不能指向空,指针可以指向空,指针初始化是记得赋空值,重载某些 ...
- MySQL外键约束On Delete、On Update各取值的含义
主键.外键和索引的区别? 主键 外键 索引 定义: 唯一标识一条记录,不能有重复的,不允许为空 表的外键是另一表的主键, 外键可以有重复的, 可以是空值 主索引(由关键字PRIMARY定义的索引) ...
- C/C++学习笔记----指针的理解
指针是C/C++编程中的重要概念之一,也是最容易产生困惑并导致程序出错的问题之一.利用指针编程可以表示各种数据结构,通过指针可使用主调函数和被调函数之间共享变量或数据结构,便于实现双向数据通讯:指针能 ...
- 转:如何学习SQL(第二部分:从关系角度理解SQL)
转自:http://blog.163.com/mig3719@126/blog/static/285720652010950825538/ 6. 从关系角度理解SQL 6.1. 关系和表 众所周知,我 ...
- C++指针理解
指针是C/C++编程中的重要概念之一,也是最容易产生困惑并导致程序出错的问题之一.利用指针编程可以表示各种数据结构,通过指针可使用主调函数和被调函数之间共享变量或数据结构,便于实现双向数据通讯:指针能 ...
- 深入理解JavaScript系列(12):变量对象(Variable Object)
介绍 JavaScript编程的时候总避免不了声明函数和变量,以成功构建我们的系统,但是解释器是如何并且在什么地方去查找这些函数和变量呢?我们引用这些对象的时候究竟发生了什么? 原始发布:Dmitry ...
- Hadoop的理解笔记
1.2Hadoop与云计算的关系1.什么是云计算:一种基于互联网的计算,在其中共享的资源.软件和信息以一种按需的方式提供给计算机和设备 , 就如同日常生活中的电网一样. 什么是Hadoop:Hadoo ...
- 20155315庄艺霖--对做中学的理解及对c语言和Java的看法
关于做中学的理解及技能训练的思考 在写这篇博客之前,我首先阅读了娄老师的博客,对做中学的概念很感兴趣.我们常说知识要学以致用,做中学强调的是在用的过程中有新的收获和体会来进一步巩固学习.细数我学过的课 ...
随机推荐
- eclipse配置Tomcat和Tomcat出现无效端口解决办法
一.eclipse配置Tomcat 1. 按图选择window-preferences 2在server处选择runtime environment . 3.点击右侧add,选择自己的Tomcat版本 ...
- clnt_create: RPC: Port mapper failure - Unable to receive: errno 113 (No route to host)
修改文件 /etc/sysconfig/nfs将#MOUNTD_PORT=892开启防火墙端口:firewalld-cmd --add-port=892/tcp
- Maven 问题 Failure to transfer org.apache.maven.plugins:maven-surefire-plugin:pom:3.0.0-M1 的处理
一.问题描述 Maven项目报错,该项目是导入的项目,然后再通过开发工具打开项目时,pom.xml文件报错. 并且新建Maven Project 也会报错. 二.报错详细Failure to tran ...
- [atAGC043B]123 Triangle
不妨先操作一轮,使得$0\le a_{i}\le 2$ 结论:若序列中存在1,则答案为0或1 考虑归纳,注意到若序列中存在1,除非所有元素均为1,否则操作一轮后必然仍存在1,那么根据归纳假设即成立,而 ...
- [hdu6997]愿望幽灵
约定:$[x^{n}]F(x)$表示多项式$F$的$n$次项系数 对于多项式$F$,定义$F$的复合逆$\hat{F}$为满足$F( ...
- [atARC115D]Odd Degree
考虑对于一棵树$G$,这个问题的答案-- 当$k$为奇数时答案显然为0,否则从$V$中任选$k$个点,以任意一点为根,从底往上不难发现子图数量唯一 换言之,当$k$为偶数时,每一个合法(恰有$k$个奇 ...
- [atARC087F]Squirrel Migration
对这棵树重心情况分类讨论: 1.若这棵树存在两个重心,分别记作$x$和$y$,如果将$(x,y)$断开,两棵子树大小都相同(都为$\frac{n}{2}$),此时$p_{i}$与$i$必然不同属于一个 ...
- Ubuntu文件权限管理
1.介绍 第一个是设备文件类型 以c开头的是字符 以b开头的是块存储 ls-l: 读写可执行 rwx | rwx | rwx 这个文件所属的用用户 组内其他成员 其他不属于用户组的成员 2.文件权限修 ...
- ICCV2021 | Swin Transformer: 使用移位窗口的分层视觉Transformer
前言 本文解读的论文是ICCV2021中的最佳论文,在短短几个月内,google scholar上有388引用次数,github上有6.1k star. 本文来自公众号CV技术指南的论文分享系 ...
- Java设计模式之(十二)——观察者模式
1.什么是观察者模式? Define a one-to-many dependency between objects so that when one object changes state, a ...