C++的智能指针

在这里插入图片描述

文章目录

  • 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.mfbz.cn/a/16570.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

axios使用笔记

文章目录 基本语法其他语法defaults config作用案例 创建实例对象作用案例 拦截器 interceptor&#xff08;AOP&#xff09;请求取消&#xff08;节流&#xff09; 基本语法 <!doctype html> <html lang"en"> <head><meta charset"UTF-8&…

可视化工作流管理

​本场景是可视化工作流&#xff0c;通过可视化的精益看板将价值流进行可视化&#xff0c;通过精益思维消除瓶颈、加速流动&#xff0c;提升效率。 创建工作流任务看板 •通过Leangoo可视化工作流项目模板&#xff0c;创建一个工作流看板。 •通过看板&#xff0c;我们可以将…

「欧拉定理」[SDOI2008]仪仗队

[SDOI2008]仪仗队 https://ac.nowcoder.com/acm/problem/20313 题目描述 作为体育委员&#xff0c;C君负责这次运动会仪仗队的训练。 仪仗队是由学生组成的N * N的方阵&#xff0c;为了保证队伍在行进中整齐划一&#xff0c;C君会跟在仪仗队的左后方&#xff0c;根据其视线所…

【计算机网络】图解内容分发网络 CDN

【计算机网络】图解内容分发网络 CDN 参考资料&#xff1a; 用了CDN就一定比不用更快吗&#xff1f; 什么是内容分发网络 高性能利器&#xff1a;CDN我建议你好好学一下&#xff01; 文章目录 【计算机网络】图解内容分发网络 CDN一、CDN 概述1.1、什么是 CDN1.2、为什么需要 …

【Java笔试强训 16】

&#x1f389;&#x1f389;&#x1f389;点进来你就是我的人了博主主页&#xff1a;&#x1f648;&#x1f648;&#x1f648;戳一戳,欢迎大佬指点! 欢迎志同道合的朋友一起加油喔&#x1f93a;&#x1f93a;&#x1f93a; 目录 一、选择题 二、判断题 &#x1f525;完全数计…

shell的基础学习三

文章目录 一、Shell 流程控制二、Shell 函数三、Shell 输入/输出重定向四、Shell 文件包含总结 一、Shell 流程控制 for 循环 与其他编程语言类似&#xff0c;Shell支持for循环。 for循环一般格式为&#xff1a; while 语句 while 循环用于不断执行一系列命令&#xff0c;也…

02-Vue技术栈之基础篇(下)

目录 1、class 与 style 绑定1.1 理解1.2 class 绑定1.3 style绑定1.4 代码示例 2、条件渲染2.1 v-if2.2 v-show2.3 注意事项2.4 代码示例 3、列表渲染3.1 基本列表3.2 key的原理3.2.1 虚拟DOM中key的作用&#xff1a;3.2.2 对比规则&#xff1a;3.2.3 用index作为key可能会引发…

IPsec中IKE与ISAKMP过程分析(主模式-消息1)

IPsec协议族中IKE&#xff08;Internet Key Exchange&#xff09;是一种基于ISAKMP的协议&#xff0c;它为建立IPSec安全通信隧道提供了一种无痕密钥交换的机制。简单来说&#xff0c;IKE就是ISAKMP的扩展&#xff0c;为ISAKMP提供了更加高效、灵活和安全的密钥协商机制。 GMT …

ChatGPT实现HTML网页文本提取

网页自动化工具 既然ChatGPT对于编程语言有非常强大的理解能力&#xff0c;那么它是否可以用来自动化地处理网页呢&#xff1f;答案是肯定的。ChatGPT可以使用机器学习算法来识别网页元素中的文本&#xff0c;并抽取出有用的信息。 例如我们提供一段层数比较多的相对来说较为…

继续科普:ChatGPT 最新写论文使用方法

这两天发现了几个国内就能用的ChatGPT,不需要魔法! 给大家推荐两种方法,大家自行选择: 1、电脑端安装VSCode软件,使用GPT插件: 优点: 无需魔法、无需付费、软件简单易用(稍懂电脑就会用) 缺点: ① 只支持电脑端,不支持手机:软件安装虽简单,但不一定所有人都…

java基础知识——22.lambda表达式

这篇文章&#xff0c;我们来讲一下java的lambda表达式 目录 1.初识lambda表达式 2.lambda表达式介绍 2.1 函数式编程 2.2 lambda表达式的具体格式 2.3 Lambda表达式的好处 2.4 Lambda的省略写法 1.初识lambda表达式 首先&#xff0c;我们来看一下lambda表达式的应用 下…

CKA/CKS/CKAD认证考试攻略

什么是CKA考试&#xff1f; CKA认证考试是由Linux基金会和云原生计算基金会(CNCF)创建的&#xff0c;以促进Kubernetes生态系统的持续发展。该考试是一种远程在线、有监考、基于实操的认证考试&#xff0c;需要在运行Kubernetes的命令行中解决多个任务。CKA认证考试是专为Kube…

SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式,系统详解springcloud微服务技术栈

Docker 我们发现在微服务中有一个令人头疼的问题——部署&#xff0c;用Docker去解决这个部署难题 &#xff08;一&#xff09;初识Docker-什么是docker 1、项目部署的问题 2、Docker 扔到一台机器上&#xff0c;它们的依赖难道没有干扰吗&#xff1f;不会&#xff0c;docke…

组合导航卡尔曼滤波几个杂项

1.组合导航卡尔曼滤波噪声协方差矩阵调参 在组合导航卡尔曼滤波算法中&#xff0c;主要涉及两个噪声协方差矩阵&#xff0c;过程噪声协方差矩阵Q&#xff0c;测量噪声协方差矩阵R&#xff0c;具体来说&#xff1a; R表示测量噪声协方差&#xff0c;它是一个数值&#xff0c;这…

低代码平台很赞,用2个小时,搭出1套应用

最近低代码很火&#xff0c;到处都是低代码的尝试贴&#xff0c;笔者今天也决定深入体验一下&#xff0c;感受它的便捷程度。 在案例开始之前&#xff0c;我们先来聊聊概念。 一、低代码 低代码实质上并不是一个新颖的话题&#xff0c;也不是最近才有的技术突破和创新&#xf…

【全年汇总】2023年CCF软件工程/系统软件/程序设计语言会议截稿时间汇总(持续更新)

本博文是根据CCF会议推荐的软件工程/系统软件/程序设计语言领域相关会议目录撰写。 一、截稿时间总览 截稿时间的总时间轴内容将会持续更新...... 往年投稿及录用情况及链接详见图片后面的内容。 二、会议详细目录 由于一些会议的投稿时间还没公开&#xff0c;因此根据往年投稿…

gRPC结合vcpkg在x64-windows平台visual studio2019 cmake工程里面的应用

这里我们运用vcpkg去下载安装gRPC&#xff0c;进入vcpkg目录后&#xff0c;执行命令&#xff1a;.\vcpkg.exe install grpc:x64-windows grpc在vcpkg里面安装完成后&#xff0c;我们就来使用grpc做一个简单的例子。 gRPC顾名思义&#xff0c;就是google的RPC方案&#xff0c;…

4月23日作业

#include <iostream> #include <cstring> using namespace std; class Student //学生类 { private: string name; //姓名 int year; //年龄 double sorce; //分数 public: Student (){} //无参构造 Student(string a,int b,double c):name(a),y…

元宇宙营销策略、玩法与案例

“元宇宙”依旧是当下品牌创新营销的重要形式&#xff0c;从时趣的行业观察来看&#xff0c;大量品牌方都有着元宇宙的营销意向&#xff0c;但在营销落地上存在不同的进度。一个显而易见的事实是&#xff0c;元宇宙不仅仅是一个虚拟的游戏空间&#xff0c;更是一个未来人人都会…

Java 抽象类和接口

一、抽象类和接口定义和使用场景 当你需要设计一些类&#xff0c;这些类有一些属性和方法是可以共享的&#xff0c;但同时又有一些属性和方法是需要不同的。在这种情况下&#xff0c;Java中提供了两种不同的机制&#xff0c;即“抽象类”和“接口”。 抽象类是一个类&#xff0…
最新文章