内容概览

一图解百问,但是有些地方我们需要特别指出:

  1. 类型在这里指通过typedef重定义的,例如函数类型、指针类型等。
  2. exception_ptr 在标准中是未定义具体实现的,因此它可能是类也可能是类型
  3. uncaught_exception()uncaught_exceptions() 是两个不同的函数,分别 判断是否有未处理异常返回未处理异常的个数 。 其中,uncaught_exceptions()函数仅在C++14开始出现。

仔细观察上图,你就能发现标准库给我们提供了以下几个方面的异常处理支持:

  • 未捕获异常处理
  • 异常嵌套
  • 异常重抛

未捕获异常处理

其实未捕获异常可以分为两类,一类有关try-catch块,另一类有关dynamic-exception-specification(动态异常指定)。

try-catch

当抛出的异常未被catch块捕获时,标准库的terminate()函数会被自动调用,默认情况下该函数调用abort()函数非正常终止程序。 非正常终止是什么意思呢? 这里我们引用标准的一句话来解释:

The program is terminated without destroying any object and without calling any of the functions passed to atexit or at_quick_exit.

大概意思是说,非正常终止不会析构任何对象,也不会调用任何通过atexit 或者 at_quick_exit注册的处理函数

很明白的,程序的资源释放会成为一个严重的问题。 因此,标准库提供了set_terminate()get_terminate()来帮助用户获取和设置处理器做一些必要的清理工作 、让用户来决定是否终止程序

dynamic-exception-specification(C++11中已废弃)

在C++11之前,函数签名中还可以指定抛出的异常类型(如果有):

void function() throw(int)	{...}

如上,如果该function函数抛出了任何非int异常类型,unexpected()函数也会被自动调用。 为此,C++标准甚至特地规定了一个bad_exception异常类来表示这种情况。 我们把这类情况也称为未捕获,该函数默认调用terminate()函数,不再赘述。

但是,但是,该特性在实践中被证明非常“鸡肋”, 因此从C++11开始被标记为废弃。

异常嵌套

为了支持异常嵌套,标准库提供了三个积木: nested_exception异常类 、throw_with_nested函数 、rethrow_if_nested函数。

如何嵌套

标准定义了一个异常类nested_exception,这个类非常特殊,它没有继承自通用的异常基类exception。 标准指出,该类是为了继承之用,以配合其它两个函数实现嵌套异常机制

如何构造嵌套

答:使用函数构造并抛出,以下是模版函数原型:

template <class T> [[noreturn]] void throw_with_nested(T&& t);

该函数将当前异常类(正在处理的异常类)与传入类型构造成一个嵌套类型, 当前异常类为nested-exception,传入类型为outer-exception。

如何解嵌套

答: 使用函数解嵌套并重抛,以下是函数原型:

template <class E> void rethrow_if_nested(const E& e);

如果传入异常类型为嵌套异常,该函数会抛出被嵌套的异常。

个人感觉

看样子,C++标准是打算提供一个方便的异常嵌套模型供开发者使用,但是给我的感觉确非常别扭。 从嵌套构造上来讲,嵌套异常的构造方式非常模糊,需要结合当前上下文,传入一个outer-exception类型也很不舒服;从解嵌套的方法上来讲,抛出似乎是比较合理的获得方式,但该函数名(rethrow_if_nested)不够友好----抛出?是抛出该嵌套异常呢还是被嵌套异常?

异常重抛

重抛的支持非常简单,调用函数:

void rethrow_exception(exception_ptr p);

但是注意,传参是exception_ptr类型,你想要重抛异常的话还需要经过一次类型转换(通过make_exception_ptr()函数)。 -……-怪不得C++总被人诟病别扭。。。

C++ 头文件系列(exception)的更多相关文章

  1. C++ 头文件系列(ios)

    1 简介 我们都知道,平时常用的那些标准流,诸如iostream.ofstream.ifstream等等,其实都是对应的basic_XXX模版的实例类. 而这些basic_XXX类模版又都是继承自同一 ...

  2. C++ 头文件系列(istream)

    1. 简介 其实叫它istream有点不合适,因为该头文件不仅定义了istream,还定义了iostream. 2. basic_istream模版 basic_istream继承自basic_ios ...

  3. C++ 头文件系列(array)

    注意,该头文件仅在C++11中标准才开始出现. 简介 与语言内置的数组一样, array类模版支持几乎所有内置数组包含的特性: 顺序的(sequence) 内存连续的(contiguous stora ...

  4. C++ 头文件系列(queue)

    简介 这个头文件定义了两个跟队列有关的类----quque.priority_queue,分别实现的是队列 和 优先队列这两个概念. 但是与这两个类模版与其它类模版(vector.array等)最大的 ...

  5. C++ 头文件系列(stack)

    简介 该头文件只含有一个类模版stack, 它实现栈的概念,是一个容器适配器(说实话,在写这篇随笔之前我都不知道有这么个类模版). 栈 栈只有一个重要的特性: LIFO(last-in first-o ...

  6. C++ 头文件系列(vector)

    简介 vector头文件包含vector的类模版以及该模版的显示特化版本vector< bool >. vector是C++容器库中非常通用的一种容器,如果你不知道该决定使用哪一种容器,或 ...

  7. C++ 头文件系列(map)

    简介 该头文件包含两个概念相似的容器----map.multimap. 而这两个容器反映的概念就是 映射. 这两个容器 相同 的属性有: 关联性 映射 动态增长 键(Key)唯一性 这两个不相同的属性 ...

  8. C++ 头文件系列(unordered_map、unordered_set)

    简介 很明显,这两个头文件分别是map.set头文件对应的unordered版本. 所以它们有一个重要的性质就是: 乱序 如何乱序 这个unorder暗示着,这两个头文件中类的底层实现----Hash ...

  9. C++ 头文件系列(iterator)

    简介 该头文件围绕迭代器展开,定义了一系列与迭代器有关的概念,但最最最重要的一点就是----它和其它容器一起实现了C++容器的Iterator设计模式. Iterators are a general ...

随机推荐

  1. LAMP学习小记

    记录今天学习到的解决LAMP环境搭建的两个小问题: 问题1.xshell无法连接到虚拟机上的linux主机 解决方法: (1)进入网络配置文件: vi /etc/sysconfig/network-s ...

  2. OSS项目进度(2.19)

    前两周先后完成了OSS.Common ,OSS.Http ,OSS.Social 项目的.net standard支持,再次开始进入框架本身的开发,今天进度情况: 一.OSS.Social 完成摇一摇 ...

  3. vim高亮设置

    vim高亮设置 前提: 查看vim是否完整安装 rpm -qa | grep vim 若完整安装,则会出现如下相关信息 vim-filesystem-7.4.629-5.el6.x86_64 vim- ...

  4. 剑指offer编程题Java实现——面试题6重建二叉树

    题目: 输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树.假设输入的前序遍历和中序遍历结果中都不含重复的数字.例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2, ...

  5. [Hadoop] - Protocol Buffer安装

    Hadoop从2.x版本开始,底层的RPC远程调用使用ProtocolBuffer格式来传递数据,所以在编译Hadoop的过程中有可能出现提示缺少Protocol服务的异常信息,类似:'protoc ...

  6. 每天一个linux命令(36)--vmstat命令

    vmstat 是 Virtual Memory Statistics(虚拟内存统计)的缩写,可对操作系统的虚拟内存.进程.CPU活动进行监控.他是对系统的整体情况进行统计,不足之处是无法对某个进程进行 ...

  7. php的empty()和isset()用法

    共同点: 1.都可以判定一个变量是否为空: 2.都返回boolean类型,即true或false. 区别: 1.isset()用来检测变量是否设置,只能用于变量,因为传递任何其它参数都将造成解析错误. ...

  8. 普实软件:MES机器数据维护

    概述 机器数据有两个菜单,机器主数据在制造数据模块下,机器MES数据相关的设置在MES模块下,两个菜单查看的内容是一致的,但是机器主数据显示的是普通的机器,可做新增.编辑.删除操作,机器MES数据仅做 ...

  9. ReactiveSwift框架

    最近项目不多,所以就研究了一下RxSwift和RAS,RAC以前项目中用过了,在这里我就先简单的介绍一下什么是RAS.总述:在RAC 5.0这个版本,有了很大的改动,API已经重新命名.在和Swift ...

  10. ajax 实现页面加载和内容的删除

    ajax最大的好处就在于加载和删除的时候不会跳转页面,现在的网页大多都会选择用ajax来写,相比嵌入PHP代码来说减少了代码量,同时加载页面也会比较快,  下面是用ajax以数据库fruit表为例写的 ...