C++中的 new / delete
new的3种形态: new operator , operator new , placement new
1.new operator: new操作符,像 + - * / && . :: ?: 等操作符一样,是语言内置的, 它 不能被重载,不能改变其行为。 它的行为包括分配内存的 operator new 和调用构造函数的 placement new。 new operator 实际上做了三件事:获得一块内存空间、调用构造函数、返回 正确的指针。如果创建的是简单类型(如char)的变量,那么第二步会被省略。
比如: CTest* pT = new CTest(1, 2);
它的调用实际上等效于:
void* p = operator new( sizeof(CTest) );
CTest* pT = new(p) CTest(2, 2);
其中前一句是operator new分配内存,后一句是placement new调用构造函 数,并返回正确的CTest*指针。
2.operator new: 操作符new,原型为: void* operator new(size_t size); 它分配指定大小的内存, 可以被重载, 可以添加额外的参数, 但第一个参数 必须为 size_t 。 它除了被 new operator 调用外也可以直接被调用, 如: void* p = operator new(sizeof(CTest)); 这种用法和调用 malloc 一样, 只分配了sizeof(CTest)大小的内存。
3.placement new: 置换new,它在一块指定的内存上调用构造函数, 包含头文件<new>之后也可 以直接使用,如: CTest* pT = new(p) CTest(2, 2); 他在p这块内存上调用CTest的构造函数来初始化CTest。 如果用 placement new 构造出来的对象,必须显示的调用对象的析构函数, 如: pT->~CTest(); 然后释放能存, 调用 operator delete (对应于分配时的 operator new) operator delete(pT); delete operator: delete操作符,和 new operator 一样,不能被重载,不能改变其行为。 delete operator 做的事有:调用析构函数,然后调用operator delete来 释放内存,比如: delete pT; 它的调用实际上等效于: pT->~CTest(); operator delete(pT); operator delete: 同理,对应于分配内存的 operator new,释放内存的为 operator delete , 它也可以被重载。 operator new [] operator delete [] 也是同样原理 ....
new 的基本使用指南
1)、想在堆上建立一个对象,应该用 new 操作符,它既分配内存又为对象调用构 造函数。
2)、如果仅仅想分配内存,就应该调用 operator new 函数;它不会调用构造函数。
3)、如果想定制在堆对象被建立时的内存分配过程,应该重载 operator new 函数, 然后使用 new operator,new operator 会调用定制的 operator new 。 4)、如果想在一块已经获得指针的内存里建立一个对象,应该用 placement new 。 placement new 主要适用于:
(a): 对时间要求非常高的应用程序中,因为这些程序分配的时间是确定的;
(b): 长时间运行而不被打断的程序;
(c): 以及执行一个垃圾收集器 (garbage collector) 。
注意:如果用 placement new 构造出来的对象,必须显示的调用对象的析构函数。
如上所述,调用 delete pT; 的执行可以看作如下这样一个过程:
pT->~CTest();
operator delete(pT); 其中: pT->~CTest() 语句调用析构函数。(析构函数本身是不会释放内存的) operator delete(pT) 释放内存。 (这是一个对象在消亡之前的最后动作) 若CTest类重载了operator delete(), 那么将不调用全局的operator delete(pT), 而是调用 pT->operator delete(pT)(在CTest::operator delete()内最好还应该调用 全局的operator delete来释放内存)。
问:成员函数调用 delete this 合法吗? 答:只要你小心,一个对象请求自杀(delete this),是可以的! 以下是对 "小心" 的定义:
1)、必须100%的确定:对象是用 new 分配的(不是用new[],也不是用 placement new,也不是一个栈上的局部对象,也不是全局的,也不 是另一个对象的成员,而是明白的普通的 new )。
2)、必须100%的确定:该成员函数是对象最后调用的的成员函数。
3)、必须100%的确定:该成员函数在 delete this 之后的代码中不接触到 对象的任何一块(包括调用任何其他成员函数或访问任何数据成员)。
4)、必须100%的确定:在 delete this 之后不再去访问this指针。你不能 去检查它,将它和其他指针比较,和NULL比较,打印它,转换它..., 不能对它做任何事。
自然,对于这种情况还要习惯性地告诫:当你的指针是一个指向基类类 型的指针,而没有虚析构函数时(也不可以delete this)。因为是在类 成员函数里面delete this的,所以在此语句以后,不能访问任何成员变 量及虚函数(调用虚函数必须对象实例存在以检查类型),否则一定非法。 上面所说的情况,在执行时不一定会报错,但尽量不要这么做。因为,一般来 说,内存释放释放的只能是数据段的内容(包括堆和栈,但释放栈上的内存由系统 进行),而代码段的内存在内存中是永远不会释放/改变的,直到程序结束。因此 在内存释放后也是可以访问的。所以,一般所谓的释放内存delete操作,是在数据 段进行的释放。但是,并不要因为这样,就违背上面的原则。
new 应用实例
/* F1 中的 new operator,他的行为就像F2中的 operator new 和 placement new 一样, 也可以用F3中的方法达到相同的效果。 */
#include <new>
#include <stdio.h>
class CTest {
public:
CTest(int _x, int _y) { X = _x; Y = _y; }
~CTest() { X = ; Y = ; }
void Test(char* sz) { printf("%s: X=%d Y=%d \n", sz, X, Y); }
int X; int Y;
}; /* new operator: */
void F1() {
CTest* pT = new CTest(, ); // new operator
pT->Test("F1");
delete pT;
} /* operator new placement new */
void F2() {
void* p = operator new(sizeof(CTest)); // operator new : 分配内存
CTest* pT = new(p) CTest(, ); // placement new: 构造对象
pT->Test("F2");
pT->~CTest(); // 必须显示析构对象
operator delete(pT); // operator delete: 释放内存
}
/* 也可这样实现: */
void F3() {
char* p = new char[sizeof(CTest)]; // new operator: char为内置类型,不会调用构造函数,相当于只分配内存
CTest* pT = new(p) CTest(, ); // placement new: 在这块内存上构造CTest对象
pT->Test("F3");
pT->~CTest(); // 必须显示析构CTest对象
delete [] p; // delete operator: char为内置类型,不会调用析构函数,相当于只释放内存
}
void main()
{
F1();
F2();
F3();
}
本文参考百度文库一篇文章,如有侵权,请联系我!
C++中的 new / delete的更多相关文章
- 数据库设计中的Soft Delete模式
最近几天有点忙,所以我们今天来一篇短的,简单地介绍一下数据库设计中的一种模式——Soft Delete. 可以说,该模式毁誉参半,甚至有非常多的人认为该模式是一个Anti-Pattern.因此在本篇文 ...
- c++中new和delete的使用方法
c++中new和delete的使用方法 new和delete运算符用于动态分配和撤销内存的运算符 new用法: 1. 开辟单变量地址空间 1)new int; //开辟一个存放数组的存储空间 ...
- C++中的new/delete与operator new/operator delete
new operator/delete operator就是new和delete操作符,而operator new/operator delete是函数. new operator(1)调用opera ...
- oracle中drop、delete和truncate的区别
oracle中drop.delete和truncate的区别 oracle中可以使用drop.delete和truncate三个命令来删除数据库中的表,网上有许多文章和教程专门讲解了它们之间的异同,我 ...
- C++中的new/delete
不同于C语言中的malloc/free是库函数,C++语言中的new/delete是运算符,而不是库函数. new/delete执行流程 我们经常会接触到的是new/delete operator(就 ...
- 数据库中drop、delete与truncate的区别
数据库中drop.delete与truncate的区别 drop直接删掉表: truncate删除表中数据,再插入时自增长id又从1开始 :delete删除表中数据,可以加where字句. (1) D ...
- SQL server触发器中 update insert delete 分别给写个例子被。
SQL server触发器中 update insert delete 分别给写个例子以及解释下例子的作用和意思被, 万分感谢!!!! 主要想知道下各个语句的书写规范. INSERT: 表1 (ID, ...
- java中File的delete()方法删除文件失败的原因
java中File的delete()方法删除文件失败的原因 学习了:http://hujinfan.iteye.com/blog/1266387 的确是忘记关闭了: 引用原文膜拜一下: 一般来说 ja ...
- SQL语句的使用,SELECT - 从数据库表中获取数据 UPDATE - 更新数据库表中的数据 DELETE - 从数据库表中删除数据 INSERT INTO - 向数据库表中插入数据
SQL DML 和 DDL 可以把 SQL 分为两个部分:数据操作语言 (DML) 和 数据定义语言 (DDL). SQL (结构化查询语言)是用于执行查询的语法. 但是 SQL 语言也包含用于更新. ...
- 浅谈 C++ 中的 new/delete 和 new[]/delete[]
在 C++ 中,你也许经常使用 new 和 delete 来动态申请和释放内存,但你可曾想过以下问题呢? new 和 delete 是函数吗? new [] 和 delete [] 又是什么?什么时候 ...
随机推荐
- IOS-NSDateFormatter使用介绍
IOS-NSDateFormatter使用介绍 NSDateFormatter的使用: NSDate *nowDate = [[NSDate alloc] init]; NSDateFormatter ...
- [转] Bound Service的三种方式(Binder、 Messenger、 AIDL)
首先要明白需要的情景,然后对三种方式进行选择: (一)可以接收Service的信息(获取Service中的方法),但不可以给Service发送信息 (二) 使用Messenger既可以接受Servic ...
- CSS块级元素、内联元素概念
CSS文档流与块级元素(block).内联元素(inline),之前翻阅不少书籍,看过不少文章, 看到所多的是零碎的CSS布局基本知识,比较表面.看过O'Reilly的<CSS权威指南>, ...
- Java基础知识强化10:Java中的中间缓存变量机制
1.对于自增运算++j与j++,由于加一的执行顺序不同,所以Java中有中间缓存变量来储存其单个表达式的值,而j的自增自减的结果依然保留在原来的变量储存区.因为本体是j的值,而单个表达式的值是中间产生 ...
- Node.js + Express + Mongodb 开发搭建个人网站(一)
一.Node + Express环境搭建 0.去Node官网下载安装node,如果安装了 npm 和 node的话 那么就 安装 全局的 express,-g全局安装 npm install expr ...
- ST3破解命令
open terminal and input it! printf '\x39' | dd seek=$((0x6f35)) conv=notrunc bs=1 of=/Applicatio ...
- java中怎么进行字符串替换?
String str = "test.doc"; String newStr = str.replaceAll("doc","html");
- 什么是DOM
什么DOM,简单的说.DOM是一套对文档的内容进行抽象和概念化的方法. 在现实世界里,人们对所谓的“世界对象模型”都不会陌生.例如,当用“汽车”.房子 和树等名词
- iOS 中如何监测某段代码运行的时间
在iOS里面有时间涉及到网络请求,有时间涉及到数据库的查询,我们需要计算该段代码的效率, 以及执行时间方面的问题,为此,可以使用下面方法: double a = CFAbsoluteTimeGetCu ...
- Java设计模式(学习整理)---策略模式
1. 模式定义 把会变化的内容取出并封装起来,以便以后可以轻易地改动或扩充部分,而不影响不需要变化的其他部分: 2.模式本质: 少用继承,多用组合,简单地说就是:固定不变的信息 ...