【C++高阶(七)】C++异常处理的方式

💓博主CSDN主页:杭电码农-NEO💓

⏩专栏分类:C++从入门到精通⏪

🚚代码仓库:NEO的学习日记🚚

🌹关注我🫵带你学习C++
  🔝🔝


在这里插入图片描述

异常处理的方式

  • 1. 前言
  • 2. C语言处理异常的方式
  • 3. C++异常概念
  • 4. 异常的抛出和匹配原则
  • 5. 异常的重新抛出
  • 6. RAII思想在异常体系中的使用
  • 7. 自定义异常体系
  • 8. C++标准库的异常体系
  • 9. 总结以及拓展

1. 前言

C++有一套独立的异常处理机制,
相信大家一定听说过try,catch这两
个词,今天就来做详细的介绍

本章重点:

本篇文章着重讲解C++异常处理的方式,
三个关键字,tyr,catch,throw,并且介绍异
常的用法和自定义体系的异常以及智能指
针在异常处理中的使用场景.其中,会复习
C语言异常处理的方式


2. C语言处理异常的方式

最经典的处理方式:使用assert

assert的缺陷:

如果在代码中使用assert,则只在debug
模式下有效,在release模式下会失效.并且
只要有错误就会直接终止程序,这明显不符
合实际,比如说在使用微信时,由于网络问题
信息没发出去,这时直接将微信程序终止了,
这样做会被乱棍打死!

C语言还能用错误码返回异常信息

错误码errno的缺陷:

返回的错误码是一个数字,程序员还需
去查表来得知这个错误码是什么意思,
并且就算查找了错误码的信息,可能它
说的不清楚,也不好看错误信息

综上所述,C语言处理异常的方式还是
不够完美,于是祖师爷写了一套自己的


3. C++异常概念

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

  • throw: 当问题出现时,程序会抛出一个异常。这是通过使用 throw 关键字来完成的
  • catch: 在您想要处理问题的地方,通过异常处理程序捕获异常.catch 关键字用于捕获异常
  • try: try 块中的代码标识将被激活的特定异常,它后面通常跟着一个或多个 catch 块。

使用方法:

try{int a,b;cin>>a>>b;if(b == 0)throw "除0错误"cout<<(a/b)<<endl;
}catch(string str)
{//......
}

如果有一个块抛出一个异常,捕获异常的方法会使用 try 和 catch 关键字。try 块中放置可能抛出异常的代码,try 块中的代码被称为保护代码,一旦throw后,会直接跳到catch的位置,后面的代码不会执行


4. 异常的抛出和匹配原则

异常的抛出和匹配有以下机制:

  1. 抛出的内容和捕捉的内容要一致

如果你在throw时抛出一个字符串,但是
在catch捕获异常时的参数却写的是整数
那么这个抛出的异常就不会去这个catch

try{int a,b;cin>>a>>b;if(b == 0)throw "除0错误"cout<<(a/b)<<endl;
}
catch(int flag)//类型与throw的不匹配,会跳到下面的catch
{//......
}
catch(string str)
{//......
}
  1. try可以嵌套多层

try和catch不仅仅可以在一个作用域
使用,还可以在最外层try,然后嵌套多
层函数,在最里面的函数throw!

void a(){throw "测试中";}
void b(){a()}
void c(){b()}
int main()
{try{c();}catch(string str){}
  1. throw和catch遵循就近原则

若写了多个catch,并且这些catch都
和throw的内容匹配,则会跳转到与
throw最近的内个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;catch (const char* errmsg) {//这个catch与throw相距最近,会优先到这儿cout << errmsg << "111" << endl;}
}
int main()
{try {Func();}catch (const char* errmsg) {cout << errmsg << "222" << endl;}return 0;
}
  1. catch(...)可以捕获任意类型的异常

在公司写大工程的时候,会和很多同事
合作写代码,大家都会抛出异常,但是你
不能确定是不是所有人抛出的类型你都
有相应的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;catch (const char* errmsg) {//这个catch与throw相距最近,会优先到这儿cout << errmsg << "111" << endl;}
}
int main()
{try {Func();}catch (const char* errmsg) {cout << errmsg << "222" << endl;}catch(...){cout<<"未知异常"<<endl;           }return 0;
}
  1. 基类可以接受抛出的子类对象

抛出和捕获有一个例外,那就是可以抛出
子类对象,用基类捕获,这个在实际场景中
非常实用,我们会在后面详谈


5. 异常的重新抛出

有可能在捕获异常时,一次捕获不能
完全解决问题,比如我们想在main函
数中处理所有的异常,在非main函数
中打印一下异常信息然后再将异常抛
到main中统一做处理

double Division(int a, int b)
{// 当b == 0时抛出异常if (b == 0){throw "Division by zero condition!";}return (double)a / (double)b;
}
void Func()
{// 这里可以看到如果发生除0错误抛出异常,另外下面的array没有得到释放。// 所以这里捕获异常后并不处理异常,异常还是交给外面处理,这里捕获了再// 重新抛出去。int* array = new int[10];try {int len, time;cin >> len >> time;cout << Division(len, time) << endl;}catch (...){cout << "delete []" << array << endl;delete[] array;throw;}cout << "delete []" << array << endl;delete[] array;
}
int main()
{try{Func();}catch (const char* errmsg){cout << errmsg << endl;}return 0;
}

在上面的场景中,如果抛出异常,就会
直接走到catch处,不会调用delete释放
释放在堆上开辟的空间,就会有问题,所
以第一次catch时先处理释放空间的问题
然后将异常再次抛出后再处理异常问题


6. RAII思想在异常体系中的使用

如果你不知道什么是RAII思想,不知道
什么是智能指针,请先阅读这篇文章:

智能指针RAII思想讲解

在异常体系中在堆上申请空间,或者
打开某个问题时常容易出问题,因为
堆上开辟的空间要显示调用delete处理
而打开的文件也要显示调用fclose关闭
所以一旦发生异常就会直接跳转到catch
的位置,有可能直接忽略了释放函数
这也就是导致了资源并没有被释放!

在异常体系中最好使用RAII思想申请资源
即使抛出异常后直接跳到catch也没问题
当出了对象作用域会自动调用析构释放!

double Division(int a, int b)
{// 当b == 0时抛出异常if (b == 0){throw "Division by zero condition!";}return (double)a / (double)b;
}
void Func()
{shared_ptr<int> array(new int(10));try {int len, time;cin >> len >> time;cout << Division(len, time) << endl;}
}
int main()
{try{Func();}catch (const char* errmsg){cout << errmsg << endl;}return 0;
}

7. 自定义异常体系

实际使用中很多公司都会自定义自己的异常体系进行规范的异常管理,因为一个项目中如果大家随意抛异常,那么外层的调用者基本就没办法玩了,所以实际中都会定义一套继承的规范体系。这样大家抛出的都是继承的派生类对象,捕获一个基类就可以了

在这里插入图片描述
不同的部分可以抛出不同的异常,然后在总的main函数中使用基类捕获所有的异常再来进行特殊的处理


8. C++标准库的异常体系

这里的内容属于了解范畴,用几张图
带大家了解一下:

在这里插入图片描述

在这里插入图片描述

实际中都是我们自己去实现一个异常体系
因为C++库做的并不好


9. 总结以及拓展

异常总体而言,利大于弊,所以工程中我们还是鼓励使用异常的。另外OO的语言基本都是用异常处理错误,这也可以看出这是大势所趋。

除此之外,异常还有一套规范,因为
比较鸡肋,所以放在了最后来介绍:

  1. 异常规格说明的目的是为了让函数使用者知道该函数可能抛出的异常有哪些。 可以在函数的后面接throw(类型),列出这个函数可能抛掷的所有异常类型。
  2. 函数的后面接throw(),表示函数不抛异常。
  3. 若无异常接口声明,则此函数可以抛掷任何类型的异常。
// 这里表示这个函数会抛出A/B/C/D中的某种类型的异常
void fun() throw(A,B,C,D);
// 这里表示这个函数只会抛出bad_alloc的异常
void* operator new (std::size_t size) throw (std::bad_alloc);
// 这里表示这个函数不会抛出异常
void* operator delete (std::size_t size, void* ptr) throw();
// C++11 中新增的noexcept,表示不会抛异常
thread() noexcept;
thread (thread&& x) noexcept;

之所以比较鸡肋是因为就算你写了
noexcept,再抛出异常
在某些编译器也不会报错


🔎 下期预告:单例模式&特殊类设计🔍

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

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

相关文章

Vim入门

Vim使用入门 1.Vim编辑器的三种常用模式 一般模式&#xff1a;刚打开文件是它&#xff0c;从编辑模式按“ESC”退回的模式也是它。可以执行各种编辑操作&#xff0c;如移动光标、复制、粘贴、删除、查找替换等 ; 编辑模式&#xff1a;在一般模式下按下 i、I、a、A、o、O 等键…

运筹系列87:julia求解随机动态规划问题入门

1. 入门案例&#xff1a;LinearPolicyGraph 看一个简单的数值优化的例子&#xff1a; 我们将其建立为一个N阶段的问题&#xff1a; 初始值为M。 使用SDDP.jl进行求解&#xff1a; using SDDP import IpoptM, N 5, 3model SDDP.LinearPolicyGraph(stages N,lower_bound …

HTTP协议在Linux系统中的运用与代码示范

在Linux系统中&#xff0c;HTTP协议的应用非常广泛&#xff0c;它被用于Web开发、网络爬虫、API调用等场景。了解并掌握HTTP协议&#xff0c;对于Linux系统的开发和使用都非常重要。下面&#xff0c;我们将为您介绍HTTP协议在Linux系统中的运用&#xff0c;并通过代码示范来帮助…

Qt for Android设置安卓程序默认横屏+全屏

我的qt版本是5.14.1&#xff0c;网上查到的方法是&#xff0c;把编译出的build文件夹中的AndroidManifest.xml文件复制出来然后修改&#xff0c;然后把修改后的xml文件加入pro文件&#xff0c;语法为ANDROID_PACKAGE_SOURCE_DIR $$PWD/AndroidManifest.xml&#xff08;具体&am…

我与Datawhale的故事之长篇

Datawhale成员 作者&#xff1a;Datawhale团队成员 前 言 上周五周年文章发出后大家反响比较热烈&#xff1a; 我们与Datawhale背后的故事&#xff01; 本期给大家带来三篇长篇回忆 胡锐峰 我与Datawhale的故事 题记&#xff1a;我和你的相遇就像春风拂面&#xff0c;就像夏雨…

【MATLAB】数据拟合第11期-基于粒子群迭代的拟合算法

有意向获取代码&#xff0c;请转文末观看代码获取方式~也可转原文链接获取~ 1 基本定义 基于粒子群迭代的拟合算法是一种优化技术&#xff0c;它基于粒子群优化算法&#xff08;PSO&#xff09;的基本思想。该算法通过群体中个体之间的协作和信息共享来寻找最优解。 在基于粒…

【数据结构】八大排序之直接插入排序算法

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:数据结构 ⚙️操作环境:Visual Studio 2022 一.直接插入排序简介及思路 直接插入排序(Straight Insertion Sort)是一种简单直观的插入排序算法. 它的基本操作是: 将一个数据插入到已经排好的有序表中,从而得到一个新的,数…

HarmonyOS--基础组件Button

Button组件 可以包含单个子组件。 Button(label?: ResourceStr, options?: { type?: ButtonType, stateEffect?: boolean }) 1&#xff1a;文字按钮 Button(‘点击’) 2&#xff1a;自定义按钮,嵌套其它组件 Button() {Image(https://) }.type(ButtonType.Circle)

强化学习------Policy Gradient算法公式推导

目录 一、前言二、公式推导基线 三、代码实现四、参考 一、前言 Policy Gradient 算法是一种基于策略的强化学习算法&#xff0c;与基于值的方法&#xff08;如Q-learning和DQN&#xff09;不同。基于值的方法主要关注于学习值函数&#xff08;如状态值函数或者动作值函数&…

Notepad++插件:格式化JSON

一、问题描述 最近有这么一串json字符串&#xff1a; 你想看吗&#xff1f; 是不是觉得密密匝匝滴&#xff0c;很不想看呢&#xff1f; 下面是经过处理的json字符串&#xff1a; 你喜欢哪种格式的json字符串展示呢&#xff1f; 反正我喜欢已经格式化的&#xff0c;也就是第二…

分享66个JavaGame源码总有一个是你想要的

分享66个JavaGame源码总有一个是你想要的 学习知识费力气&#xff0c;收集整理更不易。 知识付费甚欢喜&#xff0c;为咱码农谋福利。 游戏下载链接&#xff1a;https://pan.baidu.com/s/1BUVmun2RhAY4vAMJwcY0mQ?pwd6666 提取码&#xff1a;6666 游戏项目名称 java实现…

echarts 进度条类型柱状图

父组件&#xff1a; <barChartProfit :opt"avgProfit" />import barChartProfit from "./components/barChartProfit";data() {return {avgProfit: {xData: [2019, 2020, 2021, 2022, 2023],totalData: [30,30,30,30,30],seriesData: [10,11,12,21,…