代码编译运行环境:VS2017+Debug+Win32


异常(exception)是 C++ 引入的错误处理机制。它采用了统一的方式对程序的运行时错误进行处理,具有标准化、安全和高效的特点。C++ 为了实现异常处理,引入了三个关键字 try、throw、catch。异常由 throw 抛出,格式为 throw [expression],由catch 捕捉。try 语句块是可能抛出异常的语句块,它通常和一个或多个 catch 语句块连续出现。

try 语句块和 catch 语句块必须相互配合,以下三种情况都会导致编译错误:
(1)只有 try 语句块而没有 catch 语句块,或者只有 catch 语句块而没有 try 语句块;
(2)在 try 语句块和 catch 语句块之间夹杂其他语句;
(3)当 try 语句块后跟多个 catch 语句块时,catch 语句块之间夹杂其他语句;
(4)同一种数据类型的传值 catch 分支与传引用 catch 分支不能同时出现。

在抛出和接收异常的过程中,我们还要注意以下几点。

1.被抛出的异常对象什么时候被销毁?

用 throw 语句抛出一个对象时,会构造一个新的对象,这个对象就是异常对象。该对象的生命周期从被抛出开始计算,一直到被某个 catch 语句捕捉,就会在该 catch 语句块执行完毕后被销毁。考察如下程序。

#include <iostream>
using namespace std;class ExClass {int num;
public:ExClass(int i) {cout<<"Constructing exception object with num="<<i<<endl;num=i;}ExClass(ExClass& e) {cout<<"Copy Constructing exception object with num="<<e.num+1<<endl;num=e.num+1;}~ExClass() {cout<<"Destructing exception object with num="<<num<<endl;}void show() {cout<<"the number is "<<num<<endl;}
};int main() {ExClass obj(99);try {throw obj;     // 导致输出:Copy Constructing exception object with num=100}catch(double f) {cout<<"exception catched"<<endl;}// 导致输出:Constructing exception object with num=101catch(ExClass e) {e.show();}cout<<"after catch"<<endl;
}

程序输出结果是:

Constructing exception object with num=99
Copy Constructing exception object with num=100
Copy Constructing exception object with num=101
the number is 101
Destructing exception object with num=101
Destructing exception object with num=100
after catch
Destructing exception object with num=99

用 throw 语句抛出一个对象时,会构造一个新的对象,这个对象就是异常对象。该对象的生命周期从被抛出时开始计算,一直到被某个 catch 语句捕获,就会在该 catch 语句块执行完毕后被销毁。在上面的程序中,异常对象的 num 值为 100,“Destructing exception object with num=100” 这句话在 “after catch” 之前输出,正好说明异常对象的销毁时间是在它被捕获的 catch 块执行之后。

所有的 catch 分支在执行时类似一次函数调用,catch 的参数相当于函数的形参,而被抛出的异常对象相当于函数调用时的实参。当形参与实参成功匹配时,就说明异常被某个 catch 分支捕获。catch 后面的参数只能采用传值、传引用和传指针三种方式,如果采用传值方式,则会生成实参的一个副本,如果实参是一个对象,就会导致构造函数被调用。在上面的程序中,执行 catch(ExClass e) 语句就是利用异常对象构造一个对象 e,因此会调用拷贝构造函数。

注意:同一种数据类型的传值 catch 分支和传引用 catch 分支不能同时出现。

2.异常如果在当前函数没有被捕获会发生什么?

在某些情况下,可能所有的 catch 分支都无法捕获到抛出的异常,这将导致当前函数执行结束并返回到主调函数中。在主调函数中,将继续以上捕捉异常的过程,直到异常被捕捉或最终结束整个程序。

#include <iostream>
using namespace std;class ExClass {int num;
public:ExClass(int i) {cout<<"Constructing exception object with num="<<i<<endl;num=i;}ExClass(ExClass& e) {cout<<"Copy  Constructing exception object with num="<<e.num+1<<endl;num=e.num+1;}~ExClass() {cout<<"Destructing exception object with num="<<num<<endl;}void show() {cout<<"the number is "<<num<<endl;}
};void throwExFunc() {try {throw ExClass(199);}catch(double f) {cout<<"double exception catched"<<endl;}cout<<"exit throwExFunc()"<<endl;
}int main() {try {throwExFunc();}catch(ExClass e) {e.show();}catch(...) {cout<<"all will fall in"<<endl;}cout<<"continue to execute"<<endl;
}

程序的输出结果:

Constructing exception object with num=199
Copy Constructing exception object with num=200
the number is 200
Destructing exception object with num=200
Destructing exception object with num=199
continue to execute

从程序的结果可以看出:
(1)被抛出的异常对象的 num 值为 199,由于它没有在函数 throwExFunc() 中被捕捉,所以它导致了 throwExFunc() 的执行结束,否则会输出 exit throwExFunc()。在 main() 函数中,catch(ExClass e) 捕获了异常对象,通过复制构造函数产生对象 e,e 的num 值为 200,catch 语句块运行完结束后,对象 e 首先被销毁,紧接着销毁异常对象。在这之后,程序继续运行,输出:continue to execute。

(2)catch(…) 的意思是可以捕获所有类型的异常。不提倡随意地使用 catch(…),因为这会导致不精确处理异常类型,并降低程序的运行效率。但是,在程序的开发阶段,catch(…) 还是有用的,因为如果在精心安排异常捕获之后,还是进入了 catch(…) 语句块,说明前面的代码存在缺陷,需要进一步改正。

(3)在捕捉异常对象时,还可以采用传引用的方式,例如把 catch 语句写成 catch(ExClass& e),这样可以不必产生异常对象的副本,减少程序的运行开销,提高运行效率。

(4)在抛出异常时,还可以抛出一个指针。当然这种做法比较危险。如果要确保安全,应该将指针指向全局(静态)对象或动态申请的空间,或者被抛出的指针在本函数内被捕获。否则,利用一个被抛出的指向已经被销毁的对象指针很危险。如果实在要用,首先,必须保证对象的析构函数不能对对象的内容作破坏性修改。其次,对象的空间没有被其他新产生的变量覆盖。也就说,尽管对象被释放,但它的有效内容依然保留在栈中。


参考文献

[1] 陈刚.C++高级进阶教程[M].武汉:武汉大学出版社,2008.P365-368

C++ 抛出和接收异常的顺序相关推荐

  1. java抛出数组格式异常,Java中异常

    一.异常的概述 在Java中,把异常信息封装成了一个类.当出现了问题时,就会创建异常类对象并抛出异常相关的信息(如异常出现的位置.原因等). 二.异常的继承体系和错误的区别 1.异常的继承体系 Thr ...

  2. python输入字符a时停止_Python解释器在主提示符或从属提示符后输入中断符( )就会取消当前输入,回到主提示符,会抛出一个KeyboardInterrupt异常。...

    [单选题]下列对于字符编码的发展历史节点,正确的是( ). [单选题]Python3解释器执行 not 1 and 1的结果为( ). A. B. C. 0 D. 1 [单选题]Python3解释器执 ...

  3. java中子类可否抛出两个父类抛出的异常的子类_父类的多个构造方法各自抛出不同的异常,子类的构造方法应该抛出哪个/些异常?...

    [情况描述]如下代码,在父类中定义了两个构造方法,各自throws不同的异常.当定义子类时,IDE会提示需要显示定义构造方法来抛出父类构造方法的异常.很自然地认为应该抛出父类所有构造方法的异常,然而只 ...

  4. c# throw抛出上一个异常

    c# throw抛出上一个异常 参考文章: (1)c# throw抛出上一个异常 (2)https://www.cnblogs.com/gaara-zhang/p/9215031.html 备忘一下.

  5. paho mqtt不定时抛出Connection reset异常导致客户端掉线

    paho mqtt不定时抛出Connection reset异常导致客户端掉线 现象 分析 优化方案 现象 我们的项目采用paho的mqttv3库作客户端,起初运行状况良好,随着终端设备和用户量的增加 ...

  6. oracle异常抛出,ORACLE 存储过程异常捕获并抛出

    for tab_name in tables loop execute immediate 'drop table '||tab_name; --此处可能会报错 end loop; 当前情况是,循环表 ...

  7. java 非法参数异常_Java的比较抛出非法参数异常

    我得到这个错误:当我试图在Java中运行这个比较对我的实体系统Java的比较抛出非法参数异常 Exception in thread "Thread-3" java.lang.Il ...

  8. 怎么一次抛出多个异常

    定义一个自定义异常,如下: import java.util.ArrayList; import java.util.List;/*** 自定义异常*/ public class MyExceptio ...

  9. Python异常捕获和抛出-对方不想和你说话并向你抛出了一个异常

    文章目录 异常信息 捕获异常 多重异常处理 抛出异常 内置异常类型 自定义异常类型 异常信息 Python使用异常类来管理异常信息.当发生异常的时候,程序会抛出一个异常信息,自动根据代码的层次查找异常 ...

最新文章

  1. 华为王成录:把安卓最核心部分换得差不多了 手机升级鸿蒙OS 2.0水到渠成
  2. python变量词是什么意思_python1变量,表达式和语句
  3. 有的人在25岁时就死了,但在75岁时才被埋葬:周鸿祎
  4. 透过 3.0 Preview 看 Dubbo 的云原生变革
  5. zabbix2.4.5自带mysql监控
  6. java中static详解
  7. c matlab.h,用matlab和c写程序,include的mex.h在哪里?
  8. maven内存不足:Unexpected error occurred: Not enough memory to allocate buffers for rehashing Java heap
  9. 麒麟系统安装打印机共享_国产操作系统麒麟——文档打印 解决方案
  10. Juniper交换机配置命令_学习笔记
  11. eclipse搭建安卓开发环境
  12. 基于微信小程序的培训机构系统
  13. iOS16 系统更新教程,测试版描述文件下载
  14. JavaScript lambda 表达式介绍
  15. 整合ssh时 犯的愚蠢问题
  16. 图像风格迁移cvpr2020_CVPR 2020 论文大盘点-文本图像篇
  17. Android 中设置指定语言
  18. 通过ip反查域名信息
  19. 1.荔枝派 zero(全志V3S)-编译及SD烧录
  20. Python爬虫学习-Day7

热门文章

  1. Pwn2Own 三连冠团队成员访谈实录:如何才能登峰造极?
  2. 智慧城市数据采集的四大难点分析及解决措施
  3. Android:ListView的拓展与进阶
  4. Sentinel 简介与API订阅发布
  5. centOS6.6虚拟机启动后登陆界面无法显示
  6. while循环以及for循环的区别
  7. 三种方式实现web站点安全
  8. Linux内核list_head学习(二)
  9. Linux 下 Open××× 安装和 Windows Open××× GUI 安装笔记
  10. 上网行为审计产品对比(2008版)