C++的智能指针

news/2024/12/2 22:52:59/

在这里插入图片描述

文章目录

  • 1. 内存泄漏
    • 1.1 什么是内存泄漏
    • 1.2 内存泄漏分类
  • 2. 为什么需要智能指针
  • 3. 智能指针的使用及原理
    • 3.1 RAII
    • 3.2 使用RAII思想设计的SmartPtr类
    • 3.3 让SmartPtr像指针一样
    • 3.3 SmartPtr的拷贝
    • 3.4 auto_ptr
    • 3.5 unique_ptr
    • 3.6 shared_ptr
      • 3.6.1 shared_ptr的循环引用
      • 3.6.2 weak_ptr
    • 3.7 定制删除器

1. 内存泄漏

1.1 什么是内存泄漏

什么是内存泄漏:内存泄漏指因为疏忽或错误造成程序未能释放已经不再使用的内存的情况。内存泄漏并不是指内存在物理上的消失,而是应用程序分配某段内存后,因为设计错误,失去了对该段内存的控制,因而造成了内存的浪费。

内存泄漏的危害长期运行的程序出现内存泄漏,影响很大,如操作系统、后台服务等等,出现内存泄漏会导致响应越来越慢,最终卡死。

1.2 内存泄漏分类

C/C++程序中一般我们关心两种方面的内存泄漏:
堆内存泄漏(Heap leak):
堆内存指的是程序执行中依据须要分配通过malloc / calloc / realloc / new等从堆中分配的一块内存,用完后必须通过调用相应的 free或者delete 删掉。假设程序的设计错误导致这部分内存没有被释放,那么以后这部分空间将无法再被使用,就会产生Heap Leak。

系统资源泄漏:
指程序使用系统分配的资源,比方套接字、文件描述符、管道等没有使用对应的函数释放掉,导致系统资源的浪费,严重可导致系统效能减少,系统执行不稳定。

2. 为什么需要智能指针

我们可以采用RAII思想或者智能指针来管理资源,来避免内存泄漏。

举个例子:
在这里插入图片描述
这里有两个new出来的数组,如果new出现了异常,那么就会直接被main函数catch到。如果是第一个出现异常,可以直接被捕捉去。如果是第二个出现异常,被main里面的catch捕捉,那么第一个new就没有办法释放。久而久之,可能就会造成内存泄漏。如果我们把第二个new放到try里面,如果第二个new出现异常,被catch到,但是第二个new失败了,怎么能delete呢?所以这些都不是一个好方法,所以我们需要用到智能指针。

3. 智能指针的使用及原理

3.1 RAII

RAII是一种利用对象生命周期来控制程序资源(如内存、文件句柄、网络连接、互斥量等等)的简单技术。
在对象构造时获取资源,接着控制对资源的访问使之在对象的生命周期内始终保持有效,最后在对象析构的时候释放资源。借此,我们实际上把管理一份资源的责任托管给了一个对象

总而言之,获取到资源以后去初始化一个对象,将资料交给对象管理

这种做法有两大好处:
1.不需要显式地释放资源。
2.采用这种方式,对象所需的资源在其生命期内始终保持有效

3.2 使用RAII思想设计的SmartPtr类

在这里插入图片描述
我们这里把构造函数和析构函数先写出来。我们再去申请资源的时候就用这个类来管理:
在这里插入图片描述
在这里插入图片描述
在析构函数这里加上一个打印函数。方便我们观察。
在这里插入图片描述
不抛异常,可以正常释放。
在这里插入图片描述
抛异常也可以正常释放。如果是第一个new出现错误,直接被main里面的catch捕捉,没有事情。如果是第二个new出现错误,也会直接被捕捉,当函数栈帧销毁的时候,这个类会自动调用它的析构函数完成销毁。div()也是一样的道理,如果里面抛了除0错误,那么前面两个就会自动调用析构函数销毁。

3.3 让SmartPtr像指针一样

虽然上面管理了资源,但是我们想使用的时候就不好办了。我们需要让这个类像指针一样去使用:
在这里插入图片描述
当我们把解引用和箭头都写上时,我们就可以把这个类当指针一样去使用了。
在这里插入图片描述

3.3 SmartPtr的拷贝

举个例子:
在这里插入图片描述
因为这里是内置类型,会形成值拷贝,那么析构的时候就会析构两次,发生错误。
在这里插入图片描述

那么这里能不能使用深拷贝呢
答案是:不能的。因为智能指针的目的就是托管我们这个地值的资源,如果你深拷贝了,就不是这部分资源了,所以这里必须是浅拷贝

下面我们就说一说怎么解决这种问题。

3.4 auto_ptr

C++98版本的库中就提供了auto_ptr的智能指针。
在这里插入图片描述
文档链接 auto_ptr的实现原理:管理权转移的思想
在这里插入图片描述
这里sp1被构造出来初始化了,我们再进行下一步。
在这里插入图片描述
拷贝构造后,sp1就不能使用了。

auto_ptr拷贝构造的代码实现:
在这里插入图片描述
这些解决方法除了拷贝构造的原理不一样,其它的都和SmartPtr类一样。但是这个auto_ptr在日常中不建议使用。

3.5 unique_ptr

在C++11标准库中,出现了一些比较好的解决方法,第一个就是unique_ptr,它包含在< memory >头文件中。
在这里插入图片描述
文档链接 unique_ptr的实现原理:不让拷贝

在这里插入图片描述
它就是直接不让你拷贝了。

unique_ptr拷贝构造的代码实现:

方法一:只声明,不实现,并且声明设置成私有
在这里插入图片描述
如果我们不声明,那么编译器会默认生成一个。如果不设置成私有,那么可能会模板特化自己定义。

方法二:在C++11中,声明后加个delete
在这里插入图片描述
但是还没有防死,赋值问题没有解决:
在这里插入图片描述
我们还需要把赋值重载给写上:
在这里插入图片描述
在这里插入图片描述

3.6 shared_ptr

C++11中开始提供更靠谱的并且支持拷贝的shared_ptr。
在这里插入图片描述
文档链接 shared_ptr的实现原理:引用计数,记录几个对象管理这块资源,析构的时候- -计数,最后一个析构的对象释放资源

在这里插入图片描述
这里就没有任何问题了。

shared_ptr拷贝构造的代码实现:
那么可能有的人会想用static做一个成员变量来计数:
在这里插入图片描述
但是这样是不可以的。
在这里插入图片描述
前面两个智能指针应该里面的计数应该是2,第三个智能指针里面的计数是1。但是如果是static,它是所有这个类的对象共用一个,所以会出现问题。

正确的解决办法:
在这里插入图片描述
我们用指针来指向动态开辟的空间,当有新的智能指针时,我们就new一个。
在这里插入图片描述
拷贝构造的时候,我们将需要拷贝的对象的pCount拷贝过来++一下。
在这里插入图片描述
析构的时候,当里面为最后一个时候,把空间都给释放掉。
在这里插入图片描述
我们可以演示一下:
在这里插入图片描述
可以看到前面两个的pCount是2,第三个是1,这些都是没有问题的。

但是我们还需要处理赋值情况:
在这里插入图片描述
赋值情况就会出现问题,该加加没有加加,该减减的没有减减。

代码实现:
在这里插入图片描述
首先,自己就不需要给自己赋值了,那么我们可以这样去写:
在这里插入图片描述
不过这样写还是存在一些问题:

	sp1 = sp1;//可以避免sp1 = sp2;//不可以避免

因为sp2虽然不是和sp1同一个,但是它们的指向空间是同一个,所以第二个防不住,但是不会造成问题,只是没有作用。我们可以直接比较_ptr:
在这里插入图片描述
然后我们需要将this的对象进行减减,sp的对象进行加加。
在这里插入图片描述
这样还是不行:

	sp1 = sp3;sp2 = sp3;

如果是这种情况,那么原来sp1和sp2指向的空间没有指针去管理了,会造成内存泄漏。
在这里插入图片描述
所以我们需要把析构这个部分给加上,如果原来的pCount为0,就把这段空间销毁。

3.6.1 shared_ptr的循环引用

举个例子:
在这里插入图片描述
这是我们平时的一种简单用法,如果我们改成智能指针的话:
在这里插入图片描述
但是这里出现了问题:node1->_next和node1->_prev是原生指针,而node1和node2是一个指针指针对象,不能相互赋值。
我们需要将上面的也改成智能指针:
在这里插入图片描述
虽然编译可以通过,但是不能析构了。
在这里插入图片描述

循环引用分析
1.node1和node2两个智能指针对象指向两个节点,引用计数变成1,我们不需要手动delete
在这里插入图片描述
2. node1的_next指向node2,node2的_prev指向node1,引用计数就会变成2
在这里插入图片描述
3. node1和node2析构,引用计数减到1,但是_next还指向下一个节点。但是_prev还指向上一个节点
在这里插入图片描述
4. 此时的情况是:_next析构了,右边结点才能释放,_prev析构了,左边结点才能释放。所以这就叫循环引用,谁也不会释放

解决方法如下。

3.6.2 weak_ptr

在这里插入图片描述
文档链接 原理就是:node1->_next = node2和node2->_prev = node1时weak_ptr的_next和_prev不会增加node1和node2的引用计数

在这里插入图片描述
在这里插入图片描述
weak_ptr其实就是为了辅助shared_ptr,它不参与资源管理。

模拟实现代码:
因为weak_ptr不参与资源管理,所以它不是RAII,我们不需要pCount和析构函数:
在这里插入图片描述
最重要的是能够拷贝构造shared_ptr。那么赋值也是一样的道理。
在这里插入图片描述
我们只需要指向这段空间,不需要加计数了。

3.7 定制删除器

前面模拟实现的都是new出来的,如果是malloc或者是new []这样的方式,可能就会报错,这里设计了一个删除器来解决这个问题。

举个例子:
在这里插入图片描述
在这里插入图片描述
这里报错的原因是:delete和new []不匹配。平时我们用的unique_ptr/shared_ptr 默认释放资源用的delete

如何匹配申请方式去对应释放呢
答案是:用仿函数来做。

在这里插入图片描述
运行结果如下:
在这里插入图片描述
其它开辟空间的方式也可以用这样的方法:
在这里插入图片描述

shared_ptr虽然和unique_ptr原理类似,但是也有一点区别:
在这里插入图片描述
unique_ptr是类模板参数传参,而shared_ptr是构造函数时传参,一个传类型,一个传对象。传对象时,我们用lambda表达式比较方便。

完善unique_ptr和shared_ptr:
在这里插入图片描述

但是shared_ptr不能改造,因为标准库里实现的太复杂了。


http://www.ppmy.cn/news/58306.html

相关文章

Django入门ORM(Django操作MySQL) 专题一

Django入门ORM 原始数据库操作方式&#xff08;原生SQL&#xff09; 最早我们如果不用ORM的话&#xff0c;我们可以用MYSQL Pymysql的方式进行数据库的操作。 操作方法如下。 import pymysqldb pymysql.connect(host"", user"", password""…

现在进行时

现在进行时 解释 现在进行时态表示现在&#xff08;说话瞬间&#xff09;正在进行或者发生的动作。 例句 Look!She is reading a book.看&#xff0c;她正在读书 They are wathching TV now.他们现在正在看电视 I am doing my homework at home now.我现在正在家里做作业…

509. 斐波那契数

斐波那契数 &#xff08;通常用 F(n) 表示&#xff09;形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始&#xff0c;后面的每一项数字都是前面两项数字的和。也就是&#xff1a; F(0) 0&#xff0c;F(1) 1 F(n) F(n - 1) F(n - 2)&#xff0c;其中 n > 1 给定 n &…

二十五、OSPF高级技术——开销值、虚链路、邻居建立、LSA、静默接口

文章目录 调试指令&#xff08;三张表&#xff09;1、邻居表&#xff1a;dis ospf peer brief2、拓扑表&#xff08;链路状态数据库&#xff09;&#xff1a;dis ospf lsdb3、路由表&#xff1a;dis ip routing-table 一、OSPF 开销值/度量值&#xff08;cost&#xff09;1、co…

微信小程序学习实录1(wxml文档、引入weui、双向数据绑定、提交表单到后端)

微信小程序学习实录 一、wxml文档二、新建页面快捷方式三、微信小程序引入weui四、双向数据绑定1.wxml渲染层2.js逻辑层 提交表单到后端五、微信小程序跳转到H5 一、wxml文档 <!-- index.wxml --> <view><!-- 数据绑定 --><view><text>{{name}}…

深入理解机器学习——数据预处理:归一化 (Normalization)与标准化 (Standardization)

分类目录&#xff1a;《深入理解机器学习》总目录 归一化 &#xff08;Normalization&#xff09;和标准化 &#xff08;Standardization&#xff09;都是特征缩放的方法。特征缩放是机器学习预处理数据中最重要的步骤之一&#xff0c;可以加快梯度下降&#xff0c;也可以消除不…

osg::Drawable类通过setDrawCallback函数设置回调函数的说明

osg::Drawable类可以通过该类的setDrawCallback函数设置回调函数类对象。被设置的回调类对象必须从osg::Drawable::DrawCallback类派生&#xff0c;并重写drawImplementation函数&#xff0c;以实现自己特定的需求。这个回调函数在每次帧事件中都会被调用(如&#xff1a;在帧的…

OVS常用命令与使用总结

OVS常用命令与使用总结 说明 在平时使用ovs中&#xff0c;经常用到的ovs命令&#xff0c;参数&#xff0c;与举例总结&#xff0c;持续更新中… 进程启动 1.先准备ovs的工作目录&#xff0c;数据库存储路径等 mkdir -p /etc/openvswitch mkdir -p /var/run/openvswitch …