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

news2025/1/11 19:40:59

💓博主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错误!";
   else
   	  return ((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错误!";
   else
   	  return ((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.coloradmin.cn/o/1319393.html

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

相关文章

【密码学】群的证明(习题)

0.前置知识 1.习题 记录一次密码学作业~群的判定 2.求解

Chrome2023新版收藏栏UI改回旧版

版本 120.0.6099.109&#xff08;正式版本&#xff09;Chrome浏览器菜单新版、旧版的差异 想要将书签、功能内容改回旧版的朋友可以网址栏输入&#xff1a;「chrome://flags」&#xff0c;接着搜寻「Chrome Refresh 2023」。 最后将 Chrome Refresh 2023、Chrome Refresh 2023…

HNU-数据库系统-实验1-数据定义/数据操纵

数据库系统 课程实验1数据定义/数据操纵 计科210X 甘晴void 202108010XXX 目录 文章目录 数据库系统 课程实验1<br>数据定义/数据操纵实验目的实验样例实验环境实验内容1.1 数据库定义1&#xff09;实验内容与要求2&#xff09;实验重难点3&#xff09;实验基础知识①模…

【JAVA-Day69】抛出异常的精髓:深度解析 throw、throws 关键字,优雅处理异常问题

抛出异常的精髓&#xff1a;深度解析 throw、throws 关键字&#xff0c;优雅处理异常问题 &#x1f680; 抛出异常的精髓&#xff1a;深度解析 throw、throws 关键字&#xff0c;优雅处理异常问题 &#x1f680;一、什么是抛出异常 &#x1f60a;二、如何抛出异常 &#x1f914…

复合型下拉框

element只提供了复合型输入框&#xff0c;复合型下拉框的效果&#xff0c;我是通过button与el-select拼接形成的&#xff0c;代码如下&#xff1a; <div class"form"><button class"btn">是否需要审核</button><el-select v-model&q…

C语言之文件操作(上)

C语言之文件操作&#xff08;上&#xff09; 文章目录 C语言之文件操作&#xff08;上&#xff09;1. 什么是⽂件&#xff1f;1.1 程序⽂件1.2 数据⽂件1.3 ⽂件名 2. ⼆进制⽂件和⽂本⽂件3. ⽂件的打开和关闭3.1 流和标准流3.1.1 流3.1.2 标准流 4. ⽂件指针5. 文件的打开与关…

c语言:制造简单的计算器|练习题

一、题目 制造一个简单的计算器。输入两个数&#xff0c;计算加减乘除的结果。 如图&#xff1a; 二、代码截图【带注释】 三、源代码【带注释】 #include int main() { int yunSuanFu0;//定义运算符符号变量 int num1,num2;//定义要输入的两个数字 cc: printf(…

VUE中的8种常规通信方式

文章目录 1.props传递数据(父向子)2.$emit触发自定义事件&#xff08;子向父&#xff09;3.ref&#xff08;父子&#xff09;4.EventBus&#xff08;兄弟组件&#xff09;5.parent或root&#xff08;兄弟组件&#xff0c;有共同祖辈&#xff09;6.attrs和listeners&#xff08;…

基于Java+Swingt学生信息管理系统

基于JavaSwing学生信息管理系统 一、系统介绍二、功能展示四、其他系统实现五、获取源码 一、系统介绍 1.用户登陆&#xff1a;在帮助按钮处&#xff0c;可以查看登陆账号及密码&#xff1a; 账号admin,密码123456 在未输入的情况下&#xff0c;会提示用户名不能为空&#xff…

树专题 —— 深入理解经典红黑树

大家好&#xff0c;我是 方圆。本篇我们讲红黑树的经典实现&#xff0c;Java中对红黑树的实现便采用的是经典红黑树。前一篇文章我们介绍过左倾红黑树&#xff0c;它相对来说比较简单&#xff0c;需要大家看完上篇再来看这一篇&#xff0c;因为旋转等基础知识不会再本篇文章中赘…

程序人生15年人生感悟

计算机程序员并不是一件什么高大上的职业。而仅仅是一份普通的工作。就像医生能治病救人&#xff0c;我们能治蓝屏救程序&#xff0c;我们都在为这个世界默默的做出自己的贡献。刻意或无意宣扬某个职业高大上&#xff0c;其实质是对其它行业从业者的不公平。但是有些人却常常这…

防火墙 设置 出站规则

测试需求&#xff1a;禁止10000端口出站&#xff0c;用于测试 搜索栏 &#xff0c;输入防火墙&#xff0c;打开防火墙和网络保护&#xff0c;打开高级设置 新建一个出站规则 新建规则 &#xff0c;自定义、 这样就建好了 同时确保 防火墙&#xff0c;至少有一个打开着&#xf…

java:微服务springcloud入门以及eureka、ribbon、hystrix、feign、gateway的使用

文章目录 微服务架构Spring Cloud微服务或Spring Cloud的工作流程&#xff1a;常见注册中心技术常见的负载均衡技术常见的熔断器技术常见的配置管理技术常见的网关技术常见的消息追踪技术常见的消息总线技术示例&#xff1a;服务注册发现eureka的使用Eureka服务器服务消费者&am…

深入探索Spring Batch:大规模批处理的领航者

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏:《linux深造日志》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! ⛳️ 写在前面参与规则 ✅参与方式&#xff1a;关注博主、点赞、收藏、评论&#xff0c;任意评论&#xff08;每人最多评论…

鸿蒙 Ark ui 实战登录界面请求网络实现教程

团队介绍 作者&#xff1a;徐庆 团队&#xff1a;坚果派 公众号&#xff1a;“大前端之旅” 润开鸿生态技术专家&#xff0c;华为HDE&#xff0c;CSDN博客专家&#xff0c;CSDN超级个体&#xff0c;CSDN特邀嘉宾&#xff0c;InfoQ签约作者&#xff0c;OpenHarmony布道师&…

备赛笔记——2024全国职业院校技能大赛“大数据应用开发”赛项——任务2:离线数据处理

MySQLhttps://www.mysql.com/ 将下发的ds_db01.sql数据库文件放置mysql中 12、编写Scala代码&#xff0c;使用Spark将MySQL的ds_db01库中表user_info的全量数据抽取到Hive的ods库中表user_info。字段名称、类型不变&#xff0c;同时添加静态分区&#xff0c;分区字段为etl_da…

(1)Linux的 安装与用户的创建

前言 本章正式开始Linux的学习 如果关于Linux环境搭配有问题的朋友 可以阅读此文章:Linux环境搭建 一&#xff0c;浅用一下吧 —— Hello, Linux! 我们现在已经登陆上了&#xff0c;我们当然可以用它来做很多事。 我们来用它写一个 "Hello, Linux!" &#xff0c;来…

Mysql 计算地理位置信息

mysql 处理空间关系 MySQL提供了一系列的函数来帮助我们处理空间对象之间的关系&#xff0c;如 ST_Intersects、ST_Contains 等。这些函数可以帮助我们判断空间对象之间的位置关系&#xff0c;并在此基础上执行相应的查询。 多边形查询 在实际应用中&#xff0c;需要查询某个…

八股文打卡day2——计算机网络(2)

面试题&#xff1a;讲一下三次握手的过程&#xff1f; 我的回答&#xff1a; 1.客户端发送报文段到服务器&#xff0c;主动建立连接。这个报文段中SYN标志位表示&#xff1a;这个报文段是用于连接的&#xff0c;此时SYN标志位设置为1。其中初始序列号字段包含了客户端的初始序…

Vue 项目关于在生产环境下调试

前言 开发项目时&#xff0c;在本地调试没问题&#xff0c;但是部署到生产会遇到一些很奇怪的问题&#xff0c;本地又没法调&#xff0c;就需要在生产环境/域名下进行调试。 在这里介绍一个插件Vue force dev ,浏览器扩展里下载 即便是设置了Vue.config.devtoolsfalse 只要安…