异常和智能指针

智能指针的认识

智能指针是一种C++语言中用于管理动态内存的工具,它们可以自动管理内存的分配和释放,从而避免内存泄漏和悬空指针等问题。智能指针可以跟踪指向的对象的引用次数,并在需要时自动释放被引用的内存,这极大地提高了内存管理的安全性和便利性。

C++标准库提供了三种类型的智能指针:

  • std::auto_ptr:管理权转移的思想,当进行赋值操作时会将原对象置空,而新对象指向该空间。
     
  • std::unique_ptr:独占型智能指针,表示它所指向的对象只能被一个指针拥有(即不能进行赋值),一旦该指针销毁,它所指向的对象也会被销毁。

  • std::shared_ptr:共享型智能指针,允许多个指针共同管理同一个对象,通过引用计数来跟踪对象的引用次数,当引用计数为0时自动释放对象内存。

 其实智能指针的使用是在异常时引出的,所以我们先了解一下异常:

异常

首先我们其实在堆区new空间的时候就见过异常,当我们想要开辟的空间太大而无法申请到就会抛异常:

概念

异常是我们处理错误的一种方式,当一个函数发现自己无法处理的错误时就可以抛出异常,让函数的直接或间接的调用者处理这个错误。

  • throw: 当问题出现时,程序会抛出一个异常。这是通过使用 throw 关键字来完成的。
  • catch: 在您想要处理问题的地方,通过异常处理程序捕获异常 catch 关键字用于捕获异常,可以有多个catch进行捕获。
  • try:try 块中放置可能抛出异常的代码,它后面通常跟着一个或多个 catch 块。

使用

double Division(int a, int b)//除法计算
{// 当b == 0时抛出异常if (b == 0)throw "不能除0";elsereturn ((double)a / (double)b);
}
void Func()
{int len, time;cin >> len >> time;cout << Division(len, time) << endl;
}
int main()
{try {Func();}catch (const char* errmsg)//抛出什么类型就捕获什么类型的数据{cout << errmsg << endl;}catch (...)//捕获抛出的未知异常{cout << "unkown exception" << endl;}return 0;
}

我们要知道当我们在throw抛出异常的时候,此时就会立马调用最近的catch进行,捕获异常, 如果没有调用到catch的话,或者catch的类型不匹配的话就会出现:

而且我们抛异常之后会被立马捕获异常,而这之间的代码并不会执行,直接就开始执行catch后续代码 。 但是也别忘了在此我们调用函数的过程中是创建了栈帧的,所以我们catch捕获异常后会销毁之前的栈帧空间。

实际中抛出和捕获的匹配原则有个例外,并不都是类型完全匹配,抛出和捕获的匹配原则有个例外,并不都是类型完全匹配,可以抛出的派生类对象,使用基类捕获。这里就引入C++标准库异常体系了:

C++标准库异常体系

这个exception类就是所有标准C++异常的父类,所以当我们C++程序内部抛异常时都可以用父类对象接受捕获异常。

int main()
{try {new int[7000000000];}catch (const exception& e){cout << e.what() << endl;}catch (...){cout << "Unkown Exception" << endl;}return 0;
}


 

异常规范 

//C++11
thread() noexcept;//表示函数体内一定不会抛异常,如果抛了就运行出错
thread (thread&& x) noexcept;//C++98
void fun() throw();//表示函数体内一定不会抛异常,如果抛了也会捕获// 这里表示这个函数会抛出A/B/C/D中的某种类型的异常
void fun() throw(A,B,C,D);
// 这里表示这个函数只会抛出bad_alloc的异常
void fun() throw (std::bad_alloc);

我们是否注意到异常的一点缺陷,当抛异常后就会立马执行到捕获异常的位置,这中途的栈帧都将销毁,而代码也不会执行,但是如果这之中需要我们释放堆区空间的话,那么就会造成内存泄漏。所以我们就有了智能指针的引出:

智能指针使用与实现

我们知道catch接受异常时可能会导致从堆区动态开辟的空间未释放从而造成内存泄漏,所以我们就采用了智能指针去构造的对象,然后在栈帧销毁的时候就会自动调用析构函数了,而不需要我们进行手动调用。

int div()
{int a, b;cin >> a >> b;if (b == 0)throw invalid_argument("除0错误");return a / b;
}
void Func()
{int* p1 = new int;int* p2 = new int;cout << div() << endl;//调用div()函数delete p1;delete p2;
}
int main()
{try{Func();}catch (exception& e){cout << e.what() << endl;}return 0;
}

这里我们发现会调用div函数,然后如果抛异常的话就会造成delete函数未执行,从而导致内存泄漏的情况,因此我们就通过智能指针的方式进行改造(以shared_ptr为例):

template<class T>
class Smart_ptr
{
public:Smart_ptr(T* ptr = nullptr):_ptr(ptr){}~Smart_ptr(){delete _ptr;cout << "delete[] " << _ptr << endl;}
private:T* _ptr;
};int div()
{int a, b;cin >> a >> b;if (b == 0)throw invalid_argument("除0错误");return a / b;
}
void Func()
{Smart_ptr<int> p1 = new int;Smart_ptr<int> p2 = new int;cout << div() << endl;  
}
int main()
{try{Func();}catch (exception& e){cout << e.what() << endl;}return 0;
}

通过以上的方式,我们将需要创建的对象去构造智能指针,此时当战争销毁的时候就会自动调用类的析构函数,从而释放空间。而这种方式的原理就是就是RAII:

RAII

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

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

 shared_ptr

template<class T>
class Smart_ptr
{
public:Smart_ptr(T* ptr = nullptr):_ptr(ptr),_count(new int(1)){}Smart_ptr(const Smart_ptr<T>& tmp)//拷贝构造(浅拷贝):_ptr(tmp._ptr),_count(tmp._count){(*_count)++;//引用计数++}Smart_ptr<T>& operator=(const Smart_ptr<T>& tmp){if(_ptr!=tmp._ptr)//防止自己给自己赋值(){//this->~Smart_ptr();//没问题但是不好if (--(*_count) == 0)//该空间可能不仅被一个对象所指,所以当前的记数-1{delete[] _ptr;delete _count;//cout << "delete[] " << _ptr << endl;}_ptr = tmp._ptr;_count = tmp._count;(*_count)++;return *this;}}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}~Smart_ptr(){if (*_count > 1)(*_count)--;else{delete _ptr;delete _count;//cout << "delete[] " << _ptr << endl;}}
private:T* _ptr;int* _count;//引用计数(关键就是指针存放)//光整型存放会有问题,会导致拷贝构造后的数据不是同一个记数
};

虽然说shared_ptr可以解决这种问题,但是shared_ptr最大的缺陷就是循环引用,而对于循环引用问题的解决就引出了weak_ptr。先看一下什么是循环引用:

循环引用

循环引用是指在编程中,两个或多个对象相互引用,形成一个循环的引用链,导致它们无法被正常地回收或释放。

struct ListNode
{int val;Smart_ptr<ListNode> next;Smart_ptr<ListNode> pre;~ListNode(){cout << "~ListNode" << endl;}
};void test()
{Smart_ptr<ListNode> n1(new ListNode);Smart_ptr<ListNode> n2(new ListNode);n1->next = n2;n2->pre = n1;
}
int main()
{test();return 0;
}

在该种情形时可以分析出:n1->next=n2;这段代码使得n2指针指向的空间计数为2,而n2->pre=n1;使得n1指向的空间记数也为2,而且当该栈帧空间销毁时会先析构n2这块空间,再析构n1这块空间:

    ~Smart_ptr(){if (*_count > 1)(*_count)--;else{delete _ptr;delete _count;//cout << "delete[] " << _ptr << endl;}}

当析构n2空间时会先引用计数--,然后发现n2该对象成员函数是两个内置类型,不做处理,所以就开始析构n1对象,同样是引用计数--,此时就程序结束了。

但是_next还指向下一个节点。而且_prev还指向上一个节点。从而导致堆区内存空间未释放所以造成了内存泄漏。所以想要正确释放空间就要析构n1的_next节点(释放n2)和n2的_pre节点(释放n1),但是_next是n1的成员,_pre又是n2的成员,所以想要析构_next和_pre的话就又要析构n1和n2,从而形成了闭环。

而以上情形就是循环引用。最终谁都不会释放,从而造成内存泄漏。


std::weak_ptr是一种用于解决shared_ptr相互引用时产生死锁问题的智能指针。也就是当多个对象指向同一空间时不会增加引用计数

// 简化版的weak_ptr
template<class T>
class weak_ptr
{
public:weak_ptr():_ptr(nullptr){}weak_ptr(const shared_ptr<T>& sp):_ptr(sp.get()){}weak_ptr<T>& operator=(const shared_ptr<T>& sp){_ptr = sp.get();return *this;}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}
private:T* _ptr;
};

此时的_next和_pre 的指针类型就可以是weak_ptr的类型:


定制删除器

当我们使用智能指针的时候其实都疏忽了一点,空间释放delete对象时的方式,当我们new的是单一的空间的时候就直接delete就行了,而当我们new的是一段连续的空间的时候就需要delete[]来释放空间,此时就很容易的联想到仿函数。我们库里的是采用重载构造函数模版的方式来解决问题:

所以我们就可以按照库的解决方式进行改造我们所写的代码:

namespace cr
{template<class T>class shared_ptr{public:shared_ptr(T* ptr = nullptr):_ptr(ptr), _count(new int(1)){}template<class D>//模版类型shared_ptr(T* ptr, D del)//重载构造函数: _ptr(ptr), _count(new int(1)), _del(del)//设置为类成员用于析构调用{}shared_ptr(const shared_ptr<T>& tmp)//拷贝构造(浅拷贝):_ptr(tmp._ptr), _count(tmp._count){(*_count)++;}shared_ptr<T>& operator=(const shared_ptr<T>& tmp){if (_ptr != tmp._ptr)//防止自己给自己赋值{if (--(*_count) == 0){del(_ptr);//相当于调用operator()delete _count;}_ptr = tmp._ptr;_count = tmp._count;(*_count)++;return *this;}}~shared_ptr(){if (*_count > 1)(*_count)--;else{_del(_ptr);//相当于调用operator()delete _count;}}private:T* _ptr;int* _count;function<void(T*)> _del = [](T* ptr)->void {delete ptr; };//默认给缺省值//因为不知道_del的类型,采用function包装器包装,可以接受函数指针、lambda表达式、仿函数};
}

 主函数中进行测试调用:

template<class T>
struct del//仿函数
{void operator()(T* ptr){delete[] ptr;}
}; 
struct ListNode
{int val;cr::shared_ptr<ListNode> next;cr::shared_ptr<ListNode> pre;~ListNode(){cout << "~ListNode" << endl;}
};
int main()
{cr::shared_ptr<ListNode> p1(new ListNode[10],del<ListNode>());cr::shared_ptr<ListNode> p2(new ListNode[10],[](ListNode* ptr)->void{delete[]ptr; });cr::shared_ptr<ListNode> p3(new ListNode);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/224860.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何查看NX UI对话框内的控件(使用UIFW侦查)

一、概述 在NX二次开发中有很多命令从界面上看起开相似&#xff0c;但实质确不同&#xff0c;个人人为一是出于对软件产权的保护&#xff0c;增加二次开发的难度&#xff0c;二是由于NX在不断地发展和版本交替中为了保留老用户的操作习惯&#xff0c;故意用新控件做成老控件的…

【我与java的成长记】之面向对象的初步认识

系列文章目录 能看懂文字就能明白系列 C语言笔记传送门 &#x1f31f; 个人主页&#xff1a;古德猫宁- &#x1f308; 信念如阳光&#xff0c;照亮前行的每一步 文章目录 系列文章目录&#x1f308; *信念如阳光&#xff0c;照亮前行的每一步* 前言一、什么是面向对象面向过程…

与供应商合作:成功供应商管理的六种最佳实践

许多企业低估了他们对外部供应商的依赖程度&#xff0c;也小看了这些供应商关系所涉及的风险。本文将探索企业与外部供应商合作的六种最佳实践&#xff0c;利用它们创建有效的供应商管理流程&#xff0c;从而降低成本和风险&#xff0c;并提高盈利能力。 供应商管理为何重要&a…

【UML】第13篇 序列图(2/2)——建模的方法

目录 三、序列图建模 3.1 概述 3.2 建模的步骤 3.3 举例说明步骤 1.确定主要场景和流程 2.确定参与的对象 3.绘制序列图 4.注意事项 3.4 特殊的情况 序列图是我个人认为&#xff0c;UML中最重要的图之一。 而且序列图&#xff0c;对于业务建模&#xff0c;也有非常好…

SpringBoot + vue3 + TypeScript + activiti7 + 动态表单 的工作流引擎

工作流 SpringBoot vue3 TypeScript activiti7 动态表单 可动态绑定数据库表 的工作流引擎 介绍 兴趣使然&#xff0c;想开发一套自己认为的工作流 项目使用 vue3 elementui-plus ts Activit7 SpringBoot &#xff0c;项目从零搭建 可移植性高&#xff0c;无依赖多余…

C语言学习day10:if语句

程序流程结构&#xff1a; C 语言支持最基本的三种程序运行结构:顺序结构、选择结构、循环结构。 顺序结构:程序按顺序执行&#xff0c;不发生跳转。选择结构:依据是否满足条件&#xff0c;有选择的执行相应功能。循环结构:依据条件是否满足&#xff0c;循环多次执行某段代码…

Ubuntu20.04-查看GPU的使用情况及输出详解

1. 查看GPU的使用情况 1.1 nvidia-smi # 直接在终端得到显卡的使用情况 # 不会自动刷新 nvidia-smi# 重定向到文件中 nvidia-smi > nvidia_smi_output.txt# 如果输出的内容部分是以省略号表示的&#xff0c;可以-q nvidia-smi -q 1.2 nvidia-smi -l # 会自动刷新&#x…

eventbus,在this.$on监听事件时无法在获取数据

问题&#xff1a;vue中eventbus被多次触发&#xff0c;在this.$on监听事件时&#xff0c;内部的this发生改变导致&#xff0c;无法在vue实例中添加数据。 项目场景 一开始的需求是这样的&#xff0c;为了实现两个组件(A.vue ,B.vue)之间的数据传递。 页面A&#xff0c;点击页面…

【论文阅读笔记】SegVol: Universal and Interactive Volumetric Medical Image Segmentation

Du Y, Bai F, Huang T, et al. SegVol: Universal and Interactive Volumetric Medical Image Segmentation[J]. arXiv preprint arXiv:2311.13385, 2023.[代码开源] 【论文概述】 本文思路借鉴于自然图像分割领域的SAM&#xff0c;介绍了一种名为SegVol的先进医学图像分割模型…

C++入门编程二(各类运算符、if、switch、while、for循环等语句)

文章目录 算术运算符前后置运算符赋值运算符比较运算符逻辑运算符if语句1、单行if2、多行if3、多条件if4、嵌套if语句5、案例&#xff1a;三只小猪称体重 三目运算符switch语句while循环语句案例&#xff1a;1、猜数字 do-while循环语句案例&#xff1a;1、水仙花数 for循环语句…

Docker知识总结

文章目录 Docker1 Docker简介1.1 什么是虚拟化1.2 什么是Docker1.3 容器与虚拟机比较1.4 Docker 组件1.4.1 Docker服务器与客户端1.4.2 Docker镜像与容器1.4.3 Registry&#xff08;注册中心&#xff09; 2 Docker安装与启动2.1 安装Docker2.2 设置ustc的镜像2.3 Docker的启动与…

VMware安装linux系统二

1、设置光驱 1.1、编辑虚拟机设置 1.2、设置虚拟机镜像 1.3、设置好后开机 2、安装Linux系统 2.1、等待安装 2.2、开始安装 2.3、选择语言&#xff0c;我选择中文 2.4、本地化不用改 2.5、软件选择一定要选&#xff0c;否则就会是默认最小安装 2.6、我这里选择的是带GUI的&am…

vr虚拟高压电器三维仿真展示更立体全面

VR工业虚拟仿真软件的应用价值主要体现在以下几个方面&#xff1a; 降低成本&#xff1a;通过VR技术进行产品设计和开发&#xff0c;可以在虚拟环境中进行&#xff0c;从而减少对物理样机的依赖&#xff0c;降低试错成本和时间。此外&#xff0c;利用VR技术构建的模拟场景使用方…

北亚服务器数据恢复-服务器断电导致raid5故障的数据恢复案例

服务器数据恢复环境&#xff1a; 服务器有一组由12块硬盘组建的raid5阵列。 服务器故障&分析&#xff1a; 机房供电不稳导致服务器意外断电&#xff0c;工作人员重启服务器后发现服务器无法正常使用。 根据故障情况&#xff0c;北亚企安数据恢复工程师初步判断服务器故障原…

引用jquery.js的html5基础页面模板

本专栏是汇集了一些HTML常常被遗忘的知识&#xff0c;这里算是温故而知新&#xff0c;往往这些零碎的知识点&#xff0c;在你开发中能起到炸惊效果。我们每个人都没有过目不忘&#xff0c;过久不忘的本事&#xff0c;就让这一点点知识慢慢渗透你的脑海。 本专栏的风格是力求简洁…

MySQL索引工作原理和管理

文章目录 索引概述索引优缺点 索引结构二叉树 索引概述 MYSQL官方对索引的解释How MySQL Uses Indexes MySQL官方对索引的解释为&#xff1a;索引用于快速查找具有特定列值的行。8.3.1 How MySQL Uses IndexesIndexes are used to find rows with specific column values qui…

【深度学习-目标检测】03 - Faster R-CNN 论文学习与总结

论文地址&#xff1a;Faster R-CNN: Towards Real-Time ObjectDetection with Region Proposal Networks 论文学习 1. 摘要与引言 研究背景与挑战&#xff1a;当前最先进的目标检测网络依赖于 区域提议&#xff08;Region Proposals&#xff09;来假设目标的位置&#xff0c…

猫咪喜爱什么猫粮?五款备受赞誉的主食冻干猫粮推荐!

咱们养猫人每天最愁的就是咋给自家猫咪选一款优质的猫粮&#xff0c;让猫主子吃了健健康康的。早些年大多养猫人的标准就是盯着进口的买&#xff0c;所以之前进口猫粮的销量一直遥遥领先&#xff0c;感觉品控也严&#xff0c;也就放心大胆的冲进口猫粮了&#xff0c;但近期百利…

[Unity+文心知识库]使用百度智能云搭建私有知识库,集成知识库API,打造具备知识库的AI二次元姐姐

1.简述 最近从百度智能云的官方技术支持那边了解到&#xff0c;目前百度千帆大模型平台提供有在线的知识库功能&#xff0c;能够在线上传自己的私人知识库文档&#xff0c;并且配置文心一言模型作为文本生成的引擎&#xff0c;构建自己的私有知识库。之前自己搭建知识库都是用的…