C++ 中的异常抛出和捕获方式
作者:止步听风 发布时间:2021-06-08 21:24:19
在 C 语言中,如果发生错误,上级函数要进行出错处理,层层上传,容易造成过多的出错处理代码,并且传递的效率比较低下。
C++ 中的异常
C++ 中,异常的引发和异常的处理不必处于同一个函数中,因此底层函数可以着重于解决具体问题,而不必过多的考虑异常处理
异常是专门针对抽象编程中的一系列错误处理的,遇到错误信息就转到若干级之上进行重新尝试
异常脱离于函数机制,决定了其对函数的跨越式回跳
语法
try
{
? ? statement;
}
catch(ExceptionType var)
{
? ? statement;
}
被检测的语句放在 try 块中
try catch 语句中的花括号是语法的一部分,不能省略
try-catch 结构中,只能有一个 try 块,catch 块可以有多个,以便与不同的类型信息匹配,有点类似于 switch-case 结构
利用 throw 抛出的异常类型,可以传递系统预定义的标准类型或自定义类型
从 throw 抛出异常,到 catch 捕获异常,有点类似与利用函数的返回值进行复制一样,因此如果使用了自定义类型,需要考虑自定义类型的赋值和拷贝问题
如果 catch 语句没有与之相匹配的异常类型信息,可以用(...)表示可以捕获任何异常类型的信息,有点类似与 switch-case 结构中的 default
try-catch 结构可以与 throw 在同一函数中,也可以不在同一个函数中,throw 抛出异常后,会先在本函数中寻找与之相匹配的 catch 块,如果没有与之相匹配的 catch,就可以转到上一层 try-catch,如果仍然没有
匹配到,则转到再上一层 try-catch...,如果最终到不到与之匹配的 try-catch 块,系统就会调用系统函数,terminal 使程序终止
#include <iostream>
?
using namespace std;
?
void func1()
{
? ? double a;
? ? try{
? ? ? ? throw a;
? ? }catch(double)
? ? {
? ? ? ? cout<<"catch func1()"<<endl; //throw
? ? }
? ? cout<<"end func1()"<<endl;
? ? return ;
}
?
void func2()
{
? ? try{
? ? ? ? func1();
? ? }catch(int)
? ? {
? ? ? ? cout<<"catch func2()"<<endl;
? ? }
? ? cout<<"end func2()"<<endl;
}
?
void func3()
{
? ? try{
? ? ? ? func2();
? ? }catch(char)
? ? {
? ? ? ? cout<<"catch func3()"<<endl;
? ? }
? ? cout<<"end func3()"<<endl;
}
?
int main()
{
? ? try{
? ? ? ? func3();
? ? }catch(double)
? ? {
? ? ? ? cout<<"catch main"<<endl;
? ? }
? ? cout<<"end main"<<endl;
? ? return 0;
}
结果为:
catch func1()
end func1()
end func2()
end func3()
end main
上边的异常传递路线为 func3->func2()->func1(),在 func1 中找到对应的 catch 块,然后执行对应 catch 块中的语句,输出:
catch func1()
整个的异常处理已经结束,跳出 func1() 的 try-catch 块,继续执行 func1() 的函数体,陆续输出:
end func1()
end func2()
end func3()
end main
此时进程结束。
如果将 func1() 中的 catch 到的异常类型换个类型,如:
catch(void *)
结果为:
catch main
end main
则会在 func1(),func2(),func3() 中都找不到对应的 catch 匹配,直到 main 函数才能找到对应的匹配,然后输出:
catch main
end main
如果将 main 函数中的 catch 捕获类型也修改为:
catch(void *)
结果为:
terminate called after throwing an instance of 'double'
This application has requested the Runtime to terminate it in an unusual way.
Please contact the application's support team for more information.
此时系统就会调用系统函数,使程序终止。
抛出类型声明
为了增强程序的可读性,可以在函数声明时就列出所有可能抛出的异常类型
void func() throw (A,B,C); ? ? // 表明该函数只会抛出 A,B,C 及其子类型的异常
如果在函数声明时没有声明可能抛出的异常类型,则函数可以抛出任意类型的异常
不抛出任何类型异常的函数,可以声明为:
void func() throw();
如果一个函数抛出了抛出类型声明中所不允许的异常,unexpected 函数被调用,启用 terminal 函数中止程序
栈自旋
异常被抛出后,从进入 try 块起,到异常被抛掷前,这期间在栈上的构造的所有对象,都会被自动析构
析构的顺序与构造的顺序相反。这一过程称为栈的解旋
而堆上的空间,则会泄漏
#include <iostream>
?
using namespace std;
?
class A
{
public:
? ? A(){ cout<<"A()"<<endl; }
? ? ~A(){ cout<<"~A()"<<endl; }
};
?
int func1()
{
? ? A a;
? ? if(1)
? ? ? ? throw('a');
? ? return 0;
}
?
int func2()
{
? ? A b;
? ? func1();
? ? return 1;
}
?
int main()
{
? ? try{
? ? ? ? func2();
? ? }catch(int x){
? ? ? ? cout<<"x"<<endl;
? ? }catch(double y){
? ? ? ? cout<<"y"<<endl;
? ? }catch(...){
? ? ? ? cout<<"no x, no y"<<endl;
? ? }
? ? return 0;
}
结果为:
A()
A()
~A()
~A()
no x, no y
如果 throw 的是一个类对象:
#include <iostream>
?
using namespace std;
?
class A
{
public:
? ? A(){ cout<<"A()"<<endl; }
? ? A(const A &obj){ cout<<"A(const A &obj)"<<endl; }
? ? ~A(){ cout<<"~A()"<<endl; }
};
?
int func1()
{
? ? A a;
? ? if(1)
? ? ? ? throw(a);
? ? return 0;
}
?
int func2()
{
? ? func1();
? ? return 1;
}
?
int main()
{
? ? try{
? ? ? ? func2();
? ? }catch(int x){
? ? ? ? cout<<"x"<<endl;
? ? }catch(double y){
? ? ? ? cout<<"y"<<endl;
? ? }catch(const A &a){
? ? ? ? cout<<"no x, no y"<<endl;
? ? }
? ? return 0;
}
结果为:
A()
A(const A &obj)
~A()
no x, no y
~A()
来源:https://blog.csdn.net/SAKURASANN/article/details/105983755


猜你喜欢
- 1.给定时间戳返回指定的时间格式private string StampToDate(string timeStamp,string for
- 若要在 C++ 中实现异常处理,你可以使用 try、throw 和 catch 表达式。首先,使用 try 块将可能引发异常的一个或多个语句
- 逆时针画圆弧,原理:将360度分割成36份,分别标出每10度角度时的坐标点,然后将每个点连接起来。 #include <io
- 1.新建文件上传页面在static目录中新建upload-test.html,上传页面代码如下所示:<!DOCTYPE html>
- 写作原因:跨进程通信的实现和理解是Android进阶中重要的一环。下面博主分享IPC一些相关知识、操作及自己在学习IPC过程中的一些理解。这
- 一、ListView类1、常用的基本属性:(1)FullRowSelect:设置是否行选择模式。(默认为false) 提示:只有在Detai
- Logback日志基础配置logback日志配置有很多介绍,但是有几个非常基础的,容易忽略的。下面是最简单的一个配置,注意加粗的描述<
- 简述用来干嘛的?当你在方法中调用了多个线程,对数据库进行了一些不为人知的操作后,还有一个操作需要留到前者都执行完的重头戏,就需要用到 Cou
- 有时候一些项目并不需要提供 Web 服务,例如跑定时任务的项目,如果都按照 Web 项目启动未免画蛇添足浪费资源为了达到非 Web 运行的效
- 工厂方法模式定义: Define an interface for creating an object, but let subclass
- 本文实例为大家分享了Android实现3D层叠式卡片图片展示的具体代码,供大家参考,具体内容如下先看效果好了效果看了,感兴趣的往下看哦!整体
- 一、堆参数设置-XX:+PrintGC 使用这个参数,虚拟机启动后,只要遇到GC就会打印日志-XX:+UseSerialGC 配置串行回收器
- 本文实例为大家分享了Java流布局图形界面编写代码,供大家参考,具体内容如下package jisuanqi;import java.awt
- 限流背景在早期的计算机领域,限流技术(time limiting)被用做控制网络接口收发通信数据的速率。可以用来优化性能,减少延迟和提高带宽
- 概述本文的编写初衷,是想了解一下Spring Boot2中,具体是怎么序列化和反序列化JSR 310日期时间体系的,Spring MVC应用
- 描述符描述符是你添加到那些定义中来改变他们的意思的关键词。Java 语言有很多描述符,包括以下这些:可访问描述符不可访问描述符应用描述符,你
- java多线程-同步块Java 同步块(synchronized block)用来标记方法或者代码块是同步的。Java 同步块用来避免竞争。
- 一、运算符用于创建对象和调用构造函数。这种大家都比较熟悉,没什么好说的了。二、修饰符在用作修饰符时,new 关键字可以显式隐藏从基类继承的成
- 本文实例讲述了C#编程实现获取文件夹中所有文件的文件名。分享给大家供大家参考,具体如下:想实现这样一个功能:批量修改一个目录所有jpg文件的
- 前言CyclicBarrier和CountDownLatch这两个工具都是在java.util.concurrent包下,并且平时很多场景都