目录
1.new和delete
2.new和delete在底层是怎么实现的:
2.1new底层的实现:
2.2delete底层的实现:
2.3new []底层的实现:
2.4delete []的实现:
3.重载new和delete
4.定位new:
1.new和delete
- C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理
 - 这里在动态申请和释放时一定要匹配使用
 - 但是我们发现即使我们随意使用程序也没有崩溃,但是这些开辟的空间都是默认类型的
 - 我们接下来试一下开辟自定义类型的空间程序是否会崩溃
 - 我们运行程序,发现直接就崩溃了
 - 那为什么自定义的类型就不可以随意使用释放内存的函数呢?这就要探究malloc,free和new,delete的区别。
- 我们先来看malloc和free
 - 然后再看new和delete

2.new和delete在底层是怎么实现的:
-
2.1new底层的实现:
- 我们先来new一个test类型的空间。
 - 然后运行起来我们转到反汇编代码看看
 - 总结new的实现就是调用operate new(size_t)函数,函数内部循坏调用malloc如果申请空间成功就返回,如果申请失败就检测是否有应对措施,如果提供就执行措施,再继续malloc,如果未提供就抛出异常。然后再调用构造函数将申请的内存初始化。
-
2.2delete底层的实现:
- 我们执行delete语句,转到反汇编来
 - 进入不知名函数
 - 那么我们来看看operator delete(void *pt)函数是如何如何实现的
 - 总结:delete 的实现就是先调用类中的析构函数,将对象中的数据清空,然后调用operator delete(void *pt)函数,将内存释放掉。
-
2.3new []底层的实现: 
- 执行完这一系列操作后就会调用n次构造函数,将对象构造成功
- 总结:new []就是调用operate new[]函数,在operate new []中调用operate new函数执行刚刚的一系列操作,然后返回申请的内存。再调用n次构造函数将申请的内存初始化。
-
2.4delete []的实现: 
- 总结:delete函数就是先调用n次析构函数将申请的内存空间中的数据清空,然后再调用operate delete()函数将内存释放掉。
3.重载new和delete
- 既然有了new和delete这么好的申请内存的方法,那我们为什么还要重新实现new和delete呢?注意这里的重新实现实现new和delete并不是重新实现new和delete的申请内存的方式,而是有时我们再debug版本下调试时需要一些打印输出一些信息,这里我们需要再用new申请内存的时候要实现输出一些信息。比如说打印文件名,调用函数,调用行数等等。
- 我们重新实现new将我们的文件信息,调用函数,调用行数都打印出来
 - 可以看到虽然将函数都打印出来了但是传参的时候需要传许多参数,所以我们可以用宏替换的方式来解决
 - 但是我们需要这些信息都是在调试的时候才需要,所以我们可以条件编译一下只在debug版本底下执行
 - 我们将delete实现重载,
 - 注意这里如果我们将delete自己实现重载之后再调用delete重载函数是不会调用类中的析构函数的,如果我们的对象中涉及了资源的申请那么就不会释放。
4.定位new:
- 使用场景:在有些场景下我们可能会申请一块内存空间,但是这块内存空间并没有初始化,当我们想要使用这块内存空间时,想给他初始化,对于类类型的对象我们想要给他初始化时,只能调用构造函数初始化,但是构造函数只能在创建时由编译器自动调用(就像人不可以选择自己的出生时间),那我们这时就要使用定位new来给已经申请号的内存中创建一个对象。
 - 定位new的几种使用方式:
 - 释放空间时
 - 定位new的原理:

5.内存检测函数:_CrtDumpMemoryLeaks();
- _CrtDumpMemoryLeaks();函数是window操作系统提供的一个api(应用程序接口)函数,当程序中有内存泄漏时就会在底行输出内存泄漏信息。
 - 我们将申请的空间正确释放,可以看到没有任何的提示。

?看到这里如果觉得有用不如点个赞再走吧!!!
?
?
|