1、new

new操作针对数据类型的处理,分为两种情况:
(1)简单数据类型(包括基本数据类型和不需要构造函数的类型)
代码实例:
int* p = new int;

汇编码如下:

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

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

void *__CRTDECL operator new(size_t size) _THROW1(_STD bad_alloc){       // try to allocate size bytesvoid *p;while ((p = malloc(size)) == 0)if (_callnewh(size) == 0){       // 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 1;      //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 (1){int* p = new int[10000000];}
}

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

int* p = new int(4);

总结:

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

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

代码实例:

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

汇编码如下:

Object* p = new Object();
00AD7EDD  push        4
00AD7EDF  call        operator new (0AD1384h)
00AD7EE4  add         esp,4
00AD7EE7  mov         dword ptr [ebp-0E0h],eax
00AD7EED  mov         dword ptr [ebp-4],0
00AD7EF4  cmp         dword ptr [ebp-0E0h],0
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],0
00AD7F1A  mov         eax,dword ptr [ebp-0F4h]
00AD7F20  mov         dword ptr [ebp-0ECh],eax
00AD7F26  mov         dword ptr [ebp-4],0FFFFFFFFh
00AD7F2D  mov         ecx,dword ptr [ebp-0ECh]
00AD7F33  mov         dword ptr [p],ecx  
总结:
new 复杂数据类型的时候先调用operator new,然后在分配的内存上调用构造函数。

2、delete

delete也分为两种情况:
(1)简单数据类型(包括基本数据类型和不需要析构函数的类型)。
int *p = new int(1);
delete p;

delete的汇编码如下:

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

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

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

RTCCALLBACK默认是空的宏定义,所以这个函数默认情况下就是简单的调用free函数。
总结:
delete简单数据类型默认只是调用free函数。
(2)复杂数据类型(需要由析构函数销毁对象)
代码实例:
class Object
{
public:Object(){_val = 1;}~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+8]
012241FE  and         eax,1
01224201  je          Object::`scalar deleting destructor'+3Fh (0122420Fh)
01224203  mov         eax,dword ptr [this]
01224206  push        eax
01224207  call        operator delete (01221145h)
0122420C  add         esp,4

总结:

delete复杂数据类型先调用析构函数再调用operator delete。
3、new 数组
new[]也分为两种情况:
(1)简单数据类型(包括基本数据类型和不需要析构函数的类型)。
new[] 调用的是operator new[],计算出数组总大小之后调用operator new。
值得一提的是,可以通过()初始化数组为零值,实例:
char* p = new char[32]();

等同于:

char *p = new char[32];
memset(p, 32, 0);

总结:
针对简单类型,new[]计算好大小后调用operator new。
(2)复杂数据类型(需要由析构函数销毁对象)
实例:
class Object
{
public:Object(){_val = 1;}~Object(){cout << "destroy object" << endl;}
private:int _val;
};void main()
{Object* p = new Object[3];
}

new[]先调用operator new[]分配内存,然后在p的前四个字节写入数组大小,最后调用三次构造函数。
实际分配的内存块如下:
这里为什么要写入数组大小呢?因为对象析构时不得不用这个值,举个例子:
class Object
{
public:Object(){_val = 1;}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[3];delete[] p;
}

释放内存之前会调用每个对象的析构函数。但是编译器并不知道p实际所指对象的大小。如果没有储存数组大小,编译器如何知道该把p所指的内存分为几次来调用析构函数呢?
总结:
针对复杂类型,new[]会额外存储数组大小。
4、delete 数组
delete[]也分为两种情况:
(1)简单数据类型(包括基本数据类型和不需要析构函数的类型)。
delete和delete[]效果一样

比如下面的代码:
int* pint = new int[32];
delete pint;char* pch = new char[32];
delete pch;

运行后不会有什么问题,内存也能完成的被释放。看下汇编码就知道operator delete[]就是简单的调用operator delete。
总结:
针对简单类型,delete和delete[]等同。

(2)复杂数据类型(需要由析构函数销毁对象)
释放内存之前会先调用每个对象的析构函数。
new[]分配的内存只能由delete[]释放。如果由delete释放会崩溃,为什么会崩溃呢?
假设指针p指向new[]分配的内存。因为要4字节存储数组大小,实际分配的内存地址为[p-4],系统记录的也是这个地址。delete[]实际释放的就是p-4指向的内存。而delete会直接释放p指向的内存,这个内存根本没有被系统记录,所以会崩溃(delete指针还是delete指针指向的内存空间)。
总结:
针对复杂类型,new[]出来的内存只能由delete[]释放。
【总结】:
1、new

new操作针对数据类型的处理,分为两种情况:
(1) 简单数据类型(包括基本数据类型和不需要构造函数的类型)
  • 简单类型直接调用operator new分配内存;
  • 可以通过new_handler来处理new失败的情况;
  • new分配失败的时候不像malloc那样返回NULL,它直接抛出异常。要判断是否分配成功应该用异常捕获的机制;
(2)复杂数据类型(需要由构造函数初始化对象)
new 复杂数据类型的时候先调用operator new,然后在分配的内存上调用构造函数。
2、delete 
delete也分为两种情况:
(1) 简单数据类型(包括基本数据类型和不需要析构函数的类型)
delete简单数据类型默认只是调用free函数。
(2)复杂数据类型(需要由析构函数销毁对象)
delete复杂数据类型先调用析构函数再调用operator delete。
3、new 数组
new[]也分为两种情况:
(1) 简单数据类型(包括基本数据类型和不需要析构函数的类型)
针对简单类型,new[]计算好大小后调用operator new。
(2)复杂数据类型(需要由析构函数销毁对象)
针对复杂类型,new[]会额外存储数组大小。
4、delete 数组
delete[]也分为两种情况:
(1) 简单数据类型(包括基本数据类型和不需要析构函数的类型)
针对简单类型,delete和delete[]等同。
(2)复杂数据类型(需要由析构函数销毁对象)

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

转自:https://blog.csdn.net/passion_wu128/article/details/38966581  做略微修改

转载于:https://www.cnblogs.com/xuelisheng/p/9339957.html

【校招面试 之 C/C++】第16题 C++ new和delete的实现原理相关推荐

  1. 【校招面试 之 网络】第3题 HTTP请求行、请求头、请求体详解

    1.HTTP请求报文解剖 HTTP请求报文由3部分组成(请求行+请求头+请求体): 下面是一个实际的请求报文: ①是请求方法,GET和POST是最常见的HTTP方法,除此以外还包括DELETE.HEA ...

  2. 剑指offer 面试16题

    面试16题: 题目:数值的整数次方 题:实现函数double Power(double base, int exponent),求base的exponent次方.不得使用库函数,同时不需要考虑大数问题 ...

  3. java校招面试题_java校招面试编程题及答案.docx

    java校招面试编程题及答案 java校招面试编程题及答案 Java集合框架为Java编程语言的基础,也是Java面试中很重要的一个知识点.这里,我列出了一些关于Java集合的重要问题和答案. 集合框 ...

  4. 数字IC笔试题,大疆校招16题(仅供参考)

    \\\插播一条: 自己在今年整理一套单片机单片机相关论文800余篇 论文制作思维导图 原理图+源代码+开题报告+正文+外文资料 想要的同学私信找我. 数字IC笔试题,大疆校招16题(仅供参考) 1.异 ...

  5. 2021国家电网校招面试秘籍及真题汇总

    9月份, 按往年的政策,各省电网公司提前批校园招聘要陆续启动了. 大部分省份是在9月份发布公告, 面试集中在10月中旬左右, 但是报考上海电网.河北电网.福建电网.四川电网及新疆电网的小伙伴们注意了, ...

  6. 2023年深信服、奇安信、360等大厂网络安全校招面试真题合集(附答案)

    大家好 我是你们的学姐西米.普通本科,从事网安岗第三个年头,在北京刚刚好拿到30万出头. 其实我的技术不算特别厉害,主要是行业友好,尤其这几年明显感觉的到,网络安全愈演愈烈,很多人都开始重视网络安全 ...

  7. 校招面试——Java 基础知识

    前言 一.基本概念 1. Java程序初始化的顺序是怎么样的(B50) 2. Java和C++的区别 2. 什么是反射 3. 什么是注解 4. 什么是泛型 5. 为什么要实现内存模型? 6. 字节与字 ...

  8. 2018百度校招、腾讯校招 面试经验

    百度校招面试经验 一到九月份,就开始了各种笔试.每天晚上七点到九点,大家都可以在机房看到我一个人在那里自言自语.2018年09月16号晚上,我和我女朋友笔试完在外面吃东西时接到百度的面试通知.接到电话 ...

  9. 数据挖掘/机器学习/算法岗2017校招面试总结

    个人公众号,欢迎关注 YouChouNoBB 目前就职于腾讯,想内推朋友可以发我简历(校招/社招都要),邮箱384375530@qq.com,注明岗位和工作城市. 这个岗位叫法很多,算法岗,数据挖掘岗 ...

最新文章

  1. java命名规则_Java命名规则
  2. hdu3695(AC自动机)
  3. 使用 Flask-apidoc 自动生成 Api 文档
  4. Yii 2.0高级版 下拉框预设值、默认值
  5. Books Queries(codeforces 1066)
  6. Linux 查看文件位置/查看文件路径的命令
  7. 神经网络不收敛的 11 个原因及其解决办法
  8. mybatis快速入门(三)
  9. cheat engine 将选中目标的函数_EXCEL函数与公式剖析:IF
  10. 23. stdin, stdout, stderr
  11. matlab2013 应用程序,Matlab2013a 下载
  12. centos7安装有道词典
  13. Codeforces Round #368 (Div. 2)(C. Pythagorean Triples 勾股数规律)
  14. js 格式化prettier配置_代码格式化工具---prettier配置
  15. hazelcast_使用Hazelcast发布和订阅
  16. 韩国中产的今天!76岁的快递员,70岁的站街女......
  17. jmeter beanshell脚本使用
  18. 码科速送同城跑腿小程序V2.7.4+骑手端+前端
  19. 使用Aggregated APIServer扩展你的kubernetes API
  20. 遭遇美国TRO,原告律所撤诉后多久可以解冻?

热门文章

  1. 如何在使用新技术前评估其浏览器兼容性
  2. JDK5中的控制台输入
  3. 轻量级持久层V2版本代码与模板
  4. java去掉mongodb日志_如何禁用mongoDB java驱动程序日志记录?
  5. 白话hash和数字签名,保证你看得懂
  6. 2019web前端趋势_2019年最值得关注的Web开发趋势
  7. 计组--习题--总线
  8. (C++)CSP202009-1 称检测点查询
  9. 华为鸿蒙有机会吗,谷歌重压之下,华为鸿蒙还有机会翻盘吗?全球系统生态之争开启...
  10. oracle精度制的数据类型,ORACLE 中NUMBER 类型 低精度转换成高精度