C 语言

关注公众号 jb51net

关闭
首页 > 软件编程 > C 语言 > C++动态内存管理

一起来学习C++的动态内存管理

作者:月半木斤

这篇文章主要为大家详细介绍了C++的动态内存管理,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助

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(应用程序接口)函数,当程序中有内存泄漏时就会在底行输出内存泄漏信息。

我们将申请的空间正确释放,可以看到没有任何的提示。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注脚本之家的更多内容!

您可能感兴趣的文章:
阅读全文