C++异常特性以及使用

异常

  • 1.C传统的处理错误方式
  • 2.异常概念
  • 3.异常使用规则
    • 抛出和匹配规则
  • 4.异常的重新抛出
  • 4.异常安全
  • 5.异常规范
  • 6.使用自定义的异常
  • 7.C++标准异常体系
  • 7.异常优缺点

1.C传统的处理错误方式

  1. 终止程序:如assert,缺陷:用户难以接受。如发生内存错误,除0错误时就会终止程序。
  2. 返回错误码,缺陷:需要程序员自己去查找对应的错误。如系统的很多接口函数都是通过把错误码放到errno中,表示错误。

2.异常概念

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

  • throw:当问题出现时,程序会抛出一个异常。
  • catch:在想要处理问题的地方,通过异常处理程序捕获异常。
  • try:try块中的代码标识将被激活的特定异常,后面通常跟着一个或多个catch块。

如果有一个块抛出一个异常,捕获异常的方法会使用try和catch关键字。try块中防止可能抛出异常的代码,try块中的代码被称为保护代码。
使用格式如下:

	try{//保护代码}catch (const std::exception& e){//catch块}catch(...){//catch块}

3.异常使用规则

抛出和匹配规则

  1. 异常通过抛出对象而引发,被抛出的对象的类型,决定激活哪个catch块(类型要匹配)
  2. 匹配与该对象类型一样且离抛出异常位置最近的那一个
  3. 抛出异常对象后,会生成异常对象的拷贝,因为抛出的异常对象可能是临时对象,所以会生成一个拷贝对象,这个临时对象会在catch后被销毁(类似于传值返回)
  4. catch(…)可以捕获任意类型的异常
  5. 抛出和捕获有个例外:并不一定要完全匹配,可以抛出派生类的对象,使用基类捕获
#include<iostream>using namespace std;double Division(int a, int b)
{if (b == 0){throw "Div Zero Error!";}elsereturn double(a) / double(b);
}int main()
{int a, b;cin >> a >> b;try{cout << Division(a, b) << endl;}catch(const char* errmsg){cout << errmsg << endl;}catch (...)    //可以捕获除const char* 类型的任何数据{cout << "Unknow execption" << endl;}return 0;
}

在函数调用链中,异常展开匹配原则:
6. 首先检查throw块是否在try块内部,如果在查找匹配的catch语句。若有匹配的catch,则传到catch去处理。
7. 没有匹配的catch,就退出当前的函数栈,继续在函数栈中查找匹配的catch
8. 如果达到main函数的栈,依旧没有匹配的,则终止程序
9. 上述这个沿着调用链查找匹配的catch子句的过程称为栈展开。实际中我们最后都要加一个catch(…)捕获任意类型的异常,否则当有异常没捕获,程序就会直接终止。
10.找到匹配的catch子句并处理后,会继续沿着catch子句后面继续执行。

在这里插入图片描述

4.异常的重新抛出

该功能的目的是让最外层统一处理异常问题。

有时候单个的catch不能完全处理一个异常,在进行一些校正处理后,希望在交给更外层的调用链函数来处理,catch则可以通过重新抛出将异常传递给更上一层的函数处理。

#include<iostream>
using namespace std;double Division(int a, int b)
{if (b == 0){throw "Div Zero Error!";}elsereturn double(a) / double(b);
}void Func()
{// 这里可以看到如果发生除0错误抛出异常,另外下面的array没有得到释放。// 所以这里捕获异常后并不处理异常,异常还是交给外面处理,这里捕获了再// 重新抛出去。int a, b;cin >> a >> b;//申请数组int* c = new int[10];try{cout << Division(a, b) << endl;}catch (...){cout << "delete []" << c << endl;delete[] c;   //为了防止外面不知道怎么处理,将必要的步骤自己先处理好throw;  //再次将异常抛出去,捕到什么了抛什么}//...cout << "delete []" << c << endl;delete[] c;
}int main()
{try{Func();}catch(const char* errmsg){cout << errmsg << endl;}catch (...)    //可以捕获除const char* 类型的任何数据{cout << "Unknow execption" << endl;}return 0;
}

4.异常安全

  • 最好不要在构造函数中抛异常,否则可能会导致对象不完整或者没有完全初始化。
  • 析构函数主要完成资源的清理,最好不要在析构函数内抛异常,否则可能导致资源泄露(内存泄漏、句柄未关闭)
  • C++使用异常很容易出现问题,比如在new和delete种抛了异常,导致内存泄漏,在lock和unlock之间抛异常导致死锁,C++种经常使用RAII来解决以上问题。

5.异常规范

目的:

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

throw(),设计的不够完善,当函数真的抛异常了,不会报错。但是noexcept表示该函数不会抛异常,但是如果异常真的抛出来了,就会报错。
在这里插入图片描述

6.使用自定义的异常

在实际使用中,通常会自己定义一个异常体系,定义好基类exception后,通过继承来个性化异常。然后可以使用基类捕获子类的异常。
下面是类似的使用方法

class Exception
{
public:Exception(int errid, const string& msg):_errid(errid), _errmsg(msg){}virtual string what() const{return _errmsg;}int GetErrid() const{return _errid;}protected:int _errid;     // 错误码string _errmsg; // 错误描述
};class SqlException : public Exception
{
public:SqlException(int errid, const string& msg, const string& sql):Exception(errid, msg), _sql(sql){}virtual string what() const{string msg = "SqlException:";msg += _errmsg;msg += "->";msg += _sql;return msg;}protected:string _sql;
};class CacheException : public Exception
{
public:CacheException(const string& errmsg, int id):Exception(id, errmsg){}virtual string what() const{string msg = "CacheException:";msg += _errmsg;return msg;}
};class HttpServerException : public Exception
{
public:HttpServerException(const string& errmsg, int id, const string& type):Exception(id, errmsg), _type(type){}virtual string what() const{string msg = "HttpServerException:";msg += _errmsg;msg += "->";msg += _type;return msg;}private:const string _type;
};void SQLMgr()
{srand(time(0));if (rand() % 7 == 0){throw SqlException(100, "权限不足", "select * from name = '张三'");}cout << "调用成功" << endl;
}void CacheMgr()
{srand(time(0));if (rand() % 5 == 0){throw CacheException("权限不足", 100);}else if (rand() % 6 == 0){throw CacheException("数据不存在", 101);}SQLMgr();
}void HttpServer()
{// 模拟服务出错srand(time(0));if (rand() % 3 == 0){throw HttpServerException("请求资源不存在", 100, "get");}else if (rand() % 4 == 0){throw HttpServerException("权限不足", 101, "post");}CacheMgr();
}int main()
{while (1){this_thread::sleep_for(chrono::seconds(1));try{HttpServer();}catch (const Exception& e) // 这里捕获父类对象就可以{// 多态cout << e.what() << endl;}catch (...){cout << "Unkown Exception" << endl;}}return 0;
}

7.C++标准异常体系

在这里插入图片描述
在这里插入图片描述

7.异常优缺点

优点:

  • 异常对象定义好了,相比错误码的方式可以清晰准确的展示出各种错误信息,甚至可以包含堆栈调用信息,可以帮助程序更好的定位bug
  • 传统错误码的方式是一层一层的往上传,异常一下就传到catch位置
  • 很多第三方库,boos、gtest、gmock等等常用的库,使用它们都需要使用异常
  • 部分函数使用异常更好处理,比如构造函数没有返回值,不方便使用错误码处理。比如T& operator 这样的函数,如果pos越界了只能使用异常或者终止程序处理,没办法通过返回值表示错误。

缺点:

  • 异常会导致程序执行流乱跳,并且非常混乱,并且是运行时出错抛异常就乱跳。跟踪调试分析程序时,更困难。
  • 异常会有性能开销
  • C++没有垃圾回收机制,资源需要自己管理。异常容易导致内存泄漏、死锁等安全问题。需要使用RAII来解决。
  • C++标准体系定义不好,导致大家多使用自己定义的异常体系。
  • 异常如果不规范使用,外层捕获的用户非常痛苦,所以异常使用尽量遵守一下规范:抛出的异常都继承自一个类;函数是否抛异常、抛什么异常,都使用throw或noexception方式初始化。

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

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

相关文章

[CUDA 学习笔记] Reduce 算子优化

Reduce 算子优化 注: 本文主要是对文章 【BBuf的CUDA笔记】三&#xff0c;reduce优化入门学习笔记 - 知乎 的学习整理 Reduce 又称之为归约, 即根据数组中的每个元素得到一个输出值, 常见的包括求和(sum)、取最大值(max)、取最小值(min)等. 前言 本文同样按照英伟达官方 PP…

C++ //练习 5.6 改写上一题的程序,使用条件运算符(参见4.7节,第134页)代替if else语句。

C Primer&#xff08;第5版&#xff09; 练习 5.6 练习 5.6 改写上一题的程序&#xff0c;使用条件运算符&#xff08;参见4.7节&#xff0c;第134页&#xff09;代替if else语句。 环境&#xff1a;Linux Ubuntu&#xff08;云服务器&#xff09; 工具&#xff1a;vim 代码…

EasyRecovery软件2024永久绿色免费版下载

EasyRecovery软件在数据恢复领域享有良好的声誉&#xff0c;并且被许多用户认为是一款好用且功能强大的软件。以下是对其功能和用户评价的简要概述&#xff1a; 恢复功能&#xff1a; EasyRecovery提供全面的数据恢复功能&#xff0c;可以恢复因删除、格式化、分区丢失、硬盘故…

Ubuntu20.04更新Cmake版本详解

最近在跑一个融合惯导定位的slam框架ins_eskf_kitti&#xff0c;在框架的安装过程中&#xff0c;需要对从GitHub上克隆下来的glog进行编译。其命令如下&#xff1a; glog&#xff1a; git clone https://github.com/google/glog.git cd glog mkdir build cd build cmake .. m…

3月第一批次❗教资认定流程简易版来啦

&#x1f4dd;材料准备 1、身份证 2、户口本/学生证/居住证/毕业证 3、考试合格证明(免试人员需要师范生职业能力证书) 4、普通话证书 5、学历证书或认证报告 6、近期一寸白底证件照 7、有效期内体检报告 8、个人承诺书(网上签署) 网上申报 1、上传个人承诺书 2、校验普通话证书…

使用 C++23 从零实现 RISC-V 模拟器(2):内存和总线

&#x1f449;&#x1f3fb; 文章汇总「从零实现模拟器、操作系统、数据库、编译器…」&#xff1a;https://okaitserrj.feishu.cn/docx/R4tCdkEbsoFGnuxbho4cgW2Yntc 内存和总线 上一部分将内存全部放到了 CPU 里面&#xff0c;总线的概念是隐含着的。这一部分将内存拆分出来…

业务流程

一、需求分析和设计&#xff1a; 在项目启动阶段&#xff0c;需要与业务人员和产品经理充分沟通&#xff0c;了解业务需求&#xff0c;并根据需求进行系统设计和数据库设计。这一阶段的输出通常是需求文档、系统架构设计、数据库设计等。 1.需求文档 需求文档是一份非常重要…

java中事务的使用

文章目录 前言一、同一张表1.业务代码2.测试代码3.测试结果 二、不同表1.业务代码2.测试代码3.测试结果 总结 前言 本文将介绍在springboot中使用Transactional注解来完成对数据库事务的操作&#xff0c;保证数据一致性。 一、同一张表 1.业务代码 Controller Controller p…

【Web】CVE-2021-31805 s2-062漏洞复现学习

目录 Struts2介绍 漏洞概况 OGNL与Struts2 简单原理 漏洞复现 正向rce 反弹shell payload分析 Struts2介绍 Struts 2 是一个流行的用于构建 Java Web 应用程序的开源 Web 应用程序框架。它是 Apache 软件基金会下的一个顶级项目&#xff0c;是 Struts 框架的升级版本。…

使用 devc++ 开发 easyx 实现 Direct2D 交互

代码为 codebus 另一先生的 文案 EasyX 的三种绘图抗锯齿方法 - CodeBus 这里移植到 devc 移植操作如下&#xff1a; 调用dev 的链接库方式&#xff1a; project -> project option -> 如图所示 稍作修改的代码。 #include <graphics.h> #include <d2d1.…

计算机的大脑—中央处理单元(CPU)(重点认识运算器和控制器)

计算机的中央处理单元&#xff08;CPU&#xff09;是计算机的主要部件之一&#xff0c;负责解释和执行大部分计算机指令&#xff0c;是计算机的大脑。CPU执行的操作包括算术运算、数据传输和指令的解释执行。现代CPU的性能取决于多种因素&#xff0c;包括其核心数量、时钟速度、…

临睡之际的生死思索与生命哲学的启示

在人类生存体验中&#xff0c;有一种独特而深邃的感受——当人们准备进入梦乡时&#xff0c;会担忧第二天醒来是否还能感知到生命的律动。这种“入睡即未知”的心理状态&#xff0c;既是生命无常的深刻体现&#xff0c;也是对个体生命价值、生活态度及人生哲学的一种深度拷问。…

BUUCTF-Real-[Jupyter]notebook-rce

1、简介 Jupyter Notebook&#xff08;此前被称为 IPython notebook&#xff09;是一个交互式笔记本&#xff0c;支持运行 40 多种编程语言。 如果管理员未为Jupyter Notebook配置密码&#xff0c;将导致未授权访问漏洞&#xff0c;游客可在其中创建一个console并执行任意Pytho…

Android 10.0 锁屏壁纸 LockscreenWallpaper

前言 一、设置壁纸 通过系统设置进行锁屏壁纸和桌面壁纸的设置。 Setting 部分的代码&#xff1a; packages/apps/WallpaperPicker2/src/com/android/wallpaper/module/DefaultWallpaperPersister.java private int setStreamToWallpaperManagerCompat(InputStream inputStre…

第6个-滚动动画

Day 6 - Scroll Animation 1. 演示效果 2. 分析思路 布局 所有的内容进行水平垂直居中&#xff0c;可以使用**margin:0 auto;&#xff0c;也可以使用flex**布局&#xff1a; body {background-color: #efedd6;display: flex;flex-direction: column;justify-content: center…

树结构 严蔚敏 数据结构代码

一&#xff0c;树顺序和链式存储结构的定义 //树用儿子-兄弟表示法&#xff0c;就成了二叉树//一般二叉树用顺序存储浪费空间&#xff0c;所以大都用链式存储//特殊的二叉树有完美 或 满二叉树、完全树 可以用顺序存储//严#define MAXSIZE 100 //二叉树的最大结点数typedef TE…

前端架构: 本地调试脚手架的2种方式

一、 调试简单的脚手架方式 假定脚手架名称是 xxx 1 &#xff09;方式1 在xxx脚手架项目目录的上一级&#xff0c;执行 npm i -g xxx这时候&#xff0c;就可以本地调试脚手架&#xff0c;在前文中已经说明软链的作用参考&#xff1a;https://blog.csdn.net/Tyro_java/article…

【C语言】实现栈

目录 &#xff08;一&#xff09;栈 &#xff08;二&#xff09;头文件 &#xff08;三&#xff09;功能实现 &#xff08;1&#xff09;初始化栈 &#xff08;2&#xff09; 栈的销毁 &#xff08;3&#xff09;压栈 &#xff08;4&#xff09; 出栈 &#xff08;5&a…

【html学习笔记】1.概念

1.概念 1.1 HTML标准格式 <html><body><p>Hello World</p></body> </html>1.2 编辑方式 新建一个笔记本文件&#xff0c;将html语法格式的内容写入。保存后将记事本的.txt后缀换成.html,就可以在浏览器里运行了 1.3 中文问题 为了避…

前端架构: 从vue-cli探究脚手架原理

从使用角度理解什么是脚手架 脚手架本质是一个操作系统的客户端 在终端中去执行一个命令&#xff0c;这个命令本身它就是一个客户端我们其实可以把脚手架理解为操作系统的一个客户端通过命令去执行它的时候&#xff0c;这个命令往往是这样的一个构造&#xff0c;如下 比如&…