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
猜你喜欢
- 看到正点闹钟上的设置时间的滑动效果非常好看,自己就想做一个那样的,在网上就开始搜资料了,看到网上有的齿轮效果的代码非常多,也非常难懂,我就决
- 主要是应对这种需求:软件只允许启动一次。将这个问题转化一下,可以这样描述:对于一个软件,在启动一个进程之后,不允许启动其它进程,如果第二次打
- spring配置不扫描service层原因我将contoller给springmvc进行扫描,然后其余所有交给spring扫描然后发现ser
- 本文实例讲述了Android定时器和Handler用法。分享给大家供大家参考。具体分析如下:一、环境:主机:WIN8开发环境:Android
- 最近正好也没什么可忙的,就回过头来鼓捣过去的知识点,到Servlet部分时,以前学习的时候硬是把从上到下的继承关系和接口实现记得乱七八糟。这
- 自定义控件是根据自己的需要自己来编写控件。安卓自带的控件有时候无法满足你的需求,这种时候,我们只能去自己去实现适合项目的控件。同时,安卓也允
- 本文实例讲述了C#对图片文件的压缩、裁剪操作方法,在C#项目开发中非常有实用价值。分享给大家供大家参考。具体如下:一般在做项目时,对图片的处
- 异常的练习:老师用电脑上课。开始思考上课中出现的问题。比如问题是电脑蓝屏。电脑冒烟。要对问题进行描述,封装成对象。可是当冒烟发生后,出现讲课
- 🍊一. 为什么需要线程通信线程是并发并行的执行,表现出来是线程随机执行,但是我们在实际应用中对线程的执行顺序是有要求的,这就需要用到线程通信
- 在网上找了半天,说的都没有解决我的问题,我自己花了点时间在idea中找到并解决了问题,希望可以帮助到大家。File---->setti
- spring-retry模块支持方法和类、接口、枚举级别的重试方式很简单,引入pom包<parent> <gr
- 同步是一种只允许一个线程在特定时间访问某些资源的技术。没有其他线程可以中断,直到所分配的线程或当前访问线程访问数据完成其任务。在多线程程序中
- 如何下载并配置JDK 15进入官网下载JDK 15。官网地址:https://www.oracle.com/index.html脚本之家下载
- 本文实例讲述了C#通过流写入一行数据到文件的方法。分享给大家供大家参考。具体如下:using System;using System.IO;
- 包装类包装类其实就是8种基本数据类型对应的引用类型。基本数据类型引用数据类型byteByteshortShortintIntegerlong
- 写在前面,在笔者完成这个demo的时候,笔者发现现在大家已经不用Ajax来完成联级菜单了,实际上笔者这个demo也并不是为了完成这个,笔者主
- 我就废话不多说了,大家还是直接看代码吧~package com.jalor;import java.util.ArrayList;impor
- 本文实例讲述了C#简单输出日历的方法。分享给大家供大家参考。具体如下:用C#输出日历,此功能可用于Ajax方式列出计划日程相关的内容,由于是
- 去年买了本数字图像处理算法,一直都没有看,前几个星期都一直忙着工作上的活,趁这阶段悠闲点,玩一玩图片处理,这玩意还是非常有意思的。以前我们在
- 开发过程, 我们习惯把数据源配置, 项目常量, 日志配置等基础数据配置写到一个个单独的的文件中. 如jdbc.properties等各种.格