operator new和operator delete函数有两个重载版本:

 void* operator new (size_t);       // allocate an object
void* operator new [] (size_t); // allocate an array
void operator delete (void*); // free an oject
void operator delete [] (void*); // free an array

1、new

  new操作针对数据类型的处理,分为两种情况:

1.1 简单数据类型包括基本数据类型和不需要构造函数的类型

  代码实例:

 int* p = new int;

  汇编码如下:

     int* p = new int;
00E54C44 push
00E54C46 call operator new (0E51384h)
00E54C4B add esp,

  分析:传入4byte的参数后调用operator new。其源码如下:

void *__CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc)
{ // try to allocate size bytes
void *p;
while ((p = malloc(size)) == )
if (_callnewh(size) == )
{ // report no memory
_THROW_NCEE(_XSTD bad_alloc, );
} return (p);
}

  分析:调用malloc失败后会调用_callnewh。如果_callnewh返回0则抛出bac_alloc异常,返回非零则继续分配内存。

  这个_callnewh是什么呢?它是一个new handler,通俗来讲就是new失败的时候调用的回调函数。可以通过_set_new_handler来设置。下面举个实例:

 #include <stdio.h>
#include <new.h>
int MyNewHandler(size_t size)
{
printf("Allocation failed.Try again");
return ; //continue to allocate
//return 0; //stop allocating,throw bad_alloc
}
void main()
{
// Set the failure handler for new to be MyNewHandler.
_set_new_handler(MyNewHandler); while ()
{
int* p = new int[];
}
}

  在new基本数据类型的时候还可以指定初始化值,比如:

 int* p = new int();
  总结:
  • 简单类型直接调用operator new分配内存;
  • 可以通过new_handler来处理new失败的情况;
  • new分配失败的时候不像malloc那样返回NULL,它直接抛出异常。要判断是否分配成功应该用异常捕获的机制;

1.2 复杂数据类型(需要由构造函数初始化对象)

  代码实例:

 class Object
{
public:
Object()
{
_val = ;
} ~Object()
{
}
private:
int _val;
}; void main()
{
Object* p = new Object();
}

  汇编码如下:

     Object* p = new Object();
00AD7EDD push
00AD7EDF call operator new (0AD1384h)
00AD7EE4 add esp,
00AD7EE7 mov dword ptr [ebp-0E0h],eax
00AD7EED mov dword ptr [ebp-],
00AD7EF4 cmp dword ptr [ebp-0E0h],
00AD7EFB je main+70h (0AD7F10h)
00AD7EFD mov ecx,dword ptr [ebp-0E0h]
00AD7F03 call Object::Object (0AD1433h) //在new的地址上调用构造函数
00AD7F08 mov dword ptr [ebp-0F4h],eax
00AD7F0E jmp main+7Ah (0AD7F1Ah)
00AD7F10 mov dword ptr [ebp-0F4h],
00AD7F1A mov eax,dword ptr [ebp-0F4h]
00AD7F20 mov dword ptr [ebp-0ECh],eax
00AD7F26 mov dword ptr [ebp-],0FFFFFFFFh
00AD7F2D mov ecx,dword ptr [ebp-0ECh]
00AD7F33 mov dword ptr [p],ecx
  总结:
  new 复杂数据类型的时候先调用operator new,然后在分配的内存上调用构造函数。

2、delete

  delete也分为两种情况。

2.1 简单数据类型包括基本数据类型和不需要析构函数的类型

 int *p = new int();
delete p;

  delete的汇编码如下:

     delete p;
mov eax,dword ptr [p]
mov dword ptr [ebp-0D4h],eax
0027531D mov ecx,dword ptr [ebp-0D4h]
push ecx
call operator delete (0271127h)

  分析:传入参数p之后调用operator delete,其源码如下:

void operator delete( void * p )
{
RTCCALLBACK(_RTC_Free_hook, (p, )); free( p );
}

  RTCCALLBACK默认是空的宏定义,所以这个函数默认情况下就是简单的调用free函数。

  总结:
  delete简单数据类型默认只是调用free函数。

2.2 复杂数据类型(需要由析构函数销毁对象)

  代码实例:
 class Object
{
public:
Object()
{
_val = ;
} ~Object()
{
cout << "destroy object" << endl;
}
private:
int _val;
}; void main()
{
Object* p = new Object;
delete p;
}

  部分汇编码如下:

 012241F0  mov         dword ptr [this],ecx
012241F3 mov ecx,dword ptr [this]
012241F6 call Object::~Object (0122111Dh) //先调用析构函数
012241FB mov eax,dword ptr [ebp+]
012241FE and eax,
je Object::`scalar deleting destructor'+3Fh (0122420Fh)
mov eax,dword ptr [this]
push eax
call operator delete (01221145h)
0122420C add esp,
  总结:
  delete复杂数据类型先调用析构函数再调用operator delete。

3、new数组

3.1 简单数据类型(包括基本数据类型和不需要析构函数的类型)

  new[] 调用的是operator new[],计算出数组总大小之后调用operator new。

  值得一提的是,可以通过()初始化数组为零值,实例:

 char* p = new char[]();

  等同于:

 char *p = new char[];
memset(p, , );
  总结:
  针对简单类型,new[]计算好大小后调用operator new。

3.2 复杂数据类型(需要由析构函数销毁对象)

 class Object
{
public:
Object()
{
_val = ;
} ~Object()
{
cout << "destroy object" << endl;
}
private:
int _val;
}; void main()
{
Object* p = new Object[];
}

  new[]先调用operator new[]分配内存,然后在p的前四个字节写入数组大小,最后调用三次构造函数。

  实际分配的内存块如下:

  这里为什么要写入数组大小呢?因为对象析构时不得不用这个值,举个例子:

 class Object
{
public:
Object()
{
_val = ;
} virtual ~Object()
{
cout << "destroy Object" << endl;
}
private:
int _val;
}; class MyObject : public Object
{
public:
~MyObject()
{
cout << "destroy MyObject" << endl;
}
private:
int _foo;
}; void main()
{
Object* p = new MyObject[];
delete[] p;
}

  释放内存之前会调用每个对象的析构函数。但是编译器并不知道p实际所指对象的大小。如果没有储存数组大小,编译器如何知道该把p所指的内存分为几次来调用析构函数呢?

  总结:
  针对复杂类型,new[]会额外存储数组大小。

4、delete数组

4.1 简单数据类型(包括基本数据类型和不需要析构函数的类型)

  delete和delete[]效果一样

  比如下面的代码:

 int* pint = new int[];
delete pint; char* pch = new char[];
delete pch;

  运行后不会有什么问题,内存也能完成的被释放。看下汇编码就知道operator delete[]就是简单的调用operator delete。

  总结:
  针对简单类型,delete和delete[]等同。

4.2 复杂数据类型(需要由析构函数销毁对象)

  释放内存之前会先调用每个对象的析构函数。

  new[]分配的内存只能由delete[]释放。如果由delete释放会崩溃,为什么会崩溃呢?

  假设指针p指向new[]分配的内存。因为要4字节存储数组大小,实际分配的内存地址为[p-4],系统记录的也是这个地址。delete[]实际释放的就是p-4指向的内存。而delete会直接释放p指向的内存,这个内存根本没有被系统记录,所以会崩溃。

 
  总结:
  针对复杂类型,new[]出来的内存只能由delete[]释放。

new和delete为什么要匹配的更多相关文章

  1. GCC4.8对new和delete的参数匹配新要求

    一段通信协议的代码,早年在GCC 4.4.VS2013下编译都挺好的,移植到GCC 4.8 ,为C++ 11做准备,在编译的时候发现问题 源代码省略后的版本如下: class Zerg_App_Fra ...

  2. C++中delete[]是如何知道数组大小的

    先看一段代码: int main(void) { int *pI = new int; int *pArray = new int[10]; int size = *(pArray-1); delet ...

  3. 【M8】了解各种不同意义的new和delete

    1.首先考虑new operator,new operator 可以认为做了三件事情:a.调用operator new分配一块内存:b.在这块内存上调用构造方法构造对象:返回指针. 2.operato ...

  4. Boost汉字匹配 -- 宽字符

      原文链接:http://blog.csdn.net/sptoor/article/details/4930069 思路:汉字匹配,把字符都转换成宽字符,然后再匹配. 需要用到以下和宽字符有关的类: ...

  5. new delete

    malloc/free是标准的库函数,而new/delete是操作符 匹配使用原则:malloc(calloc/realloc)和free 以及new/new[] 和delete/delete[]; ...

  6. C#进阶系列——WebApi 路由机制剖析:你准备好了吗?

    前言:从MVC到WebApi,路由机制一直是伴随着这些技术的一个重要组成部分. 它可以很简单:如果你仅仅只需要会用一些简单的路由,如/Home/Index,那么你只需要配置一个默认路由就能简单搞定: ...

  7. shell example02

    输入值 //相加 add(){ echo "add two agrs..." echo "enter first one: " read arg1 echo & ...

  8. MySQL连接线程kill利器之pt-kill

    如何每10秒检查一次,杀死指定用户超过100秒的查询? pt-kill \ --no-version-check \ --host 127.0.0.1 --port 3306 --user 'xxxx ...

  9. asp.net MVC4的执行流程

    MVC在底层和传统的asp.net是一致的,在底层之上,相关流程如下: 1)Global.asax里,MvcApplication对象的Application_Start()事件中,调用 RouteC ...

随机推荐

  1. JS中遍历普通数组和字典数组的区别

    // 普通数组 var intArray = new Array(); intArray[0] = "第一个"; intArray[1] = "第二个"; fo ...

  2. Otto Product Classification Winner's Interview: 2nd place, Alexander Guschin ¯\_(ツ)_/¯

    Otto Product Classification Winner's Interview: 2nd place, Alexander Guschin ¯\_(ツ)_/¯ The Otto Grou ...

  3. 团体程序设计天梯赛-练习集L2-005. 集合相似度

    L2-005. 集合相似度 时间限制 400 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 给定两个整数集合,它们的相似度定义为:Nc/Nt*1 ...

  4. smarty 时间格式化date_format

    代码如下:$smarty = new Smarty; $smarty->assign('yesterday', strtotime('-1 day')); $smarty->display ...

  5. [笨木头FireFly 02]入门篇2_客户端发送请求,服务器处理请求

    原地址:http://www.9miao.com/question-15-53940.html 好,经过上一篇不权威的讲解,大家已经能轻易地让客户端和服务端连接起来了. 但是,仅仅是连接了,可它们俩不 ...

  6. php继承与重载

    <?php class A { public $param = "paramA"; public function test() { echo "testA&quo ...

  7. c++ 孟岩推荐 书籍

    c++ primer 中文版本  是 教程+参考书 扛梁之作c++ 标准程序库 对于c++熟手来说更为快捷effective c++  永远是初学者必读的,但是c++11标准后的第四版,还未发布c++ ...

  8. 12232 - Exclusive-OR

    12232 - Exclusive-OR 题目大意是可以设定一个点Xp=v,或者Xp^Xq=v,然后查询Xa^Xb^Xc...等于多少. 由于异或操作跟判连通性很类似,这里可以使用并查集来解决,对于X ...

  9. 条件与(&&)和逻辑与(&)以及条件或(||)和逻辑或(|)区别

    条件与(&&)和逻辑与(&)以及条件或(||)和逻辑或(|)区别在于它们的运算结果是不相同的. 条件与(&&)和条件或(||)采用的是所谓的"短路规则 ...

  10. 【HDOJ】4122 Alice's mooncake shop

    RMQ的基础题目,简单题. /* 4122 */ #include <iostream> #include <sstream> #include <string> ...