异常处理

一、相关资料

【C++】异常简述(一):C语言中的异常处理机制

【C++】异常简述(二):C++的异常处理机制

【C++】异常简述(三):补充之如何看待C++异常

[ 之后,针对本章节、根据链接再进行一次系统的学习 ]

基本结构

try {
throw logic_error{"blah"};
} catch (exception) {
// caught here!

} catch (logic_error) {
// not here!
}

百家讲坛 - "必要性"

[1]

为什么需要异常机制:https://blog.csdn.net/K346K346/java/article/details/50087193

C++之父Bjarne Stroustrup在《The C++ Programming Language》中讲到:

(a) 一个库的作者可以检测出发生了运行时错误,但一般不知道怎样去处理它们(因为和用户具体的应用有关);

(b) 另一方面,库的用户知道怎样处理这些错误,但却无法检查它们何时发生(如果能检测,就可以在用户的代码里处理了,不用留给库去发现)。

Bjarne Stroustrup说:提供异常的基本目的就是为了处理上面的问题。基本思想是:让一个函数在发现了自己无法处理的错误时抛出(throw)一个异常,然后它的(直接或者间接)调用者能够处理这个问题。

[2]

C++ 引入异常的原因之一是:为了能让构造函数报错(析构函数不能抛异常这是大家都知道的常识),毕竟构造函数没有返回值,没有异常的话调用方如何得知对象构造是否成功呢?

[3]

对使用 C++ 异常处理应具有怎样的态度?

到了异常,一般就直接让本次操作失效,保存状态好了.

定义异常

一、内置 Error类型

下表是对上面层次结构中出现的每个异常的说明:

异常 描述
std::exception 该异常是所有标准 C++ 异常的父类。
std::bad_alloc 该异常可以通过 new 抛出。
std::bad_cast 该异常可以通过 dynamic_cast 抛出。
std::bad_exception 这在处理 C++ 程序中无法预期的异常时非常有用。
std::bad_typeid 该异常可以通过 typeid 抛出。
std::logic_error 理论上可以通过读取代码来检测到的异常。
- std::domain_error 当使用了一个无效的数学域时,会抛出该异常。
- std::invalid_argument 当使用了无效的参数时,会抛出该异常。
- std::length_error 当创建了太长的 std::string 时,会抛出该异常。
- std::out_of_range 该异常可以通过方法抛出,例如 std::vector 和 std::bitset<>::operator[]()。
std::runtime_error 理论上不可以通过读取代码来检测到的异常。
- std::overflow_error 当发生数学上溢时,会抛出该异常。
- std::range_error 当尝试存储超出范围的值时,会抛出该异常。
- std::underflow_error 当发生数学下溢时,会抛出该异常。

二、自定义异常类型

通过继承和重载 exception 类来定义新的异常。

#include <iostream>
#include <exception>
using namespace std;

// 竟然是个struct
struct MyException : public exception
{
const char* what () const throw
{
return "C++ Exception";
}
}; // -------------------------------------------------
int main()
{
try
{
throw MyException();
}
catch(MyException& e)
{
std::cout << "MyException caught" << std::endl;
std::cout << e.what() << std::endl;
}
catch(std::exception& e)
{
// 其他的错误
}
}

类可以继承,结构体也可以继承.

#include <iostream>  
using namespace std;

//
class MyException : public exception
{
public:
const char* what() const throw ()   // <----
{
std::cout << "my exception" << std::endl;
return NULL;
}
}; void f1(bool flag = true)
{
if (flag) throw MyException();
} void f2(bool flag = true) throw ()
{
if (flag) throw MyException();
}

----------------------------------------------------- int main(void)
{
try
{
f1();
}
catch (...)
{
;
}
std::cout << "f1()异常将被捕获,不会 abort,将继续执行" << std::endl; try
{
f2();
}
catch (...)
{
;
}
std::cout << "f2()异常不会被捕获,程序将会 abort,将不会执行该条语句" << std::endl; return ;
}

使用异常

Catch-by-reference is BETTER

Problems with Catch-by-value:

    • inefficient due to object copying
    • causes the slicing problem, and
    • cannot exploit polymorphism and dynamic binding

Catch-by-reference avoids all these problems.

catch的参数是引用,自然就不需要拷贝了。

  • Catch-by-Value

#include <iostream>

using namespace std;

class X {
public:
X() {
std::cout << "X constructed" << std::endl;
} X(const X &x) {
std::cout << "X copy-constructed" << std::endl;
} ~X() {
std::cout << "X destructed" << std::endl;
}
};

void g() {
throw X{};
} void f() {
try {
g();
} catch (X x) {  // 改为catch(X &x), 其他同理
std::cout << "caught in f; rethrow" << std::endl;
throw;
}
}

int main()
{
cout << "Hello World!" << endl; try {
f();
} catch (X x) {  // 这里又是一次 浪费时间的 copy
std::cout << "caught in main" << std::endl;
} return ;
}

Result:

  • Catch-by-Reference

改为reference后,concise了不少!

  • Stack Unwinding

关于new object这件事上,Obversely, this avoids memory leaks but is messy.

便有了Smart Pointer? 这里是初始化列表的用法。

Stack Exception Safety

如果noexcept修饰的函数抛出了异常,编译器可以选择直接调用std::terminate()函数来终止程序的运行,这比基于异常机制的throw()在效率上会高一些。

这是因为异常机制会带来一些额外开销,比如函数抛出异常,会导致函数栈被依次地展开(unwind),并依帧调用在本帧中已构造的自动变量的析构函数等。

Common Levels of Exception Safety

End.

[c++] Exceptions的更多相关文章

  1. "NHibernate.Exceptions.GenericADOException: could not load an entity" 解决方案

     今天,测试一个项目的时候,抛出了这个莫名其妙的异常,然后就开始了一天的调试之旅... 花了很长时间,没有从代码找出任何问题... 那么到底哪里出问题呢? 根据下面那段长长的错误日志: -- ::, ...

  2. 2000条你应知的WPF小姿势 基础篇<34-39 Unhandled Exceptions和Resource>

    在正文开始之前需要介绍一个人:Sean Sexton. 来自明尼苏达双城的软件工程师.最为出色的是他维护了两个博客:2,000Things You Should Know About C# 和 2,0 ...

  3. com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: Unknown column 'dd' in 'where clause'

    今天在使用mysql数据库查找数据的时候报错,错误信息如下: com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: Unknown co ...

  4. MIT 6.828 JOS学习笔记18. Lab 3.2 Part B: Page Faults, Breakpoints Exceptions, and System Calls

    现在你的操作系统内核已经具备一定的异常处理能力了,在这部分实验中,我们将会进一步完善它,使它能够处理不同类型的中断/异常. Handling Page Fault 缺页中断是一个非常重要的中断,因为我 ...

  5. Solve VS2010 Error "Exceptions has been thrown by the target of an invocation"

    Sometimes when you open a VS2010 project, an error window will pop up with the error message "E ...

  6. MyBatis-Exception:org.apache.ibatis.exceptions.PersistenceException

    错误信息如下: HTTP Status 500 - org.mybatis.spring.MyBatisSystemException: nested exception is org.apache. ...

  7. App Transport Security has blocked a cleartext HTTP (http://) resource load since it is insecure. Temporary exceptions can be configured via your app's Info.plist file

    ios进行http请求,会出现这个问题: App Transport Security has blocked a cleartext HTTP (http://) resource load sin ...

  8. 添加 All Exceptions 断点后, 每次运行都会在 main.m 中断的一种解决方法

    在本人项目添加导入和使用新的字体过程中,遇到一个很奇怪的问题: 项目开启了全局断点,但是每次启动都会运行在mian.m中断,点击下一步程序继续正常运行. 不知道是什么原因,于是google百度寻找答案 ...

  9. python基于Django框架编译报错“django.core.exceptions.ImproperlyConfigured”的解决办法?

    下面是我具体遇到的问题和解决方法: 错误详细信息: django.core.exceptions.ImproperlyConfigured: Requested setting DEFAULT_IND ...

  10. Rethrowing exceptions and preserving the full call stack trace

    refer:http://weblogs.asp.net/fmarguerie/archive/2008/01/02/rethrowing-exceptions-and-preserving-the- ...

随机推荐

  1. Swing学习笔记1-----Swing组件类的层次

    1.  从结构上划分 Swing 组件类分为两种,一种是JComponent类,一种是Windows类.其中windows类包含的是一些可以独立显示的组件,而JComponent类包含的是不可以独立显 ...

  2. MySql UDF 调用外部程序和系统命令

    1.mysql利用mysqludf的一个mysql插件可以实现调用外部程序和系统命令 下载lib_mysqludf_sys程序:https://github.com/mysqludf/lib_mysq ...

  3. 最小生成树 prime poj1287

    poj1287 裸最小生成树 代码 #include "map" #include "queue" #include "math.h" #i ...

  4. 【转】Polya定理

    转自:http://endlesscount.blog.163.com/blog/static/82119787201221324524202/ Polya定理 首先记Sn为有前n个正整数组成的集合, ...

  5. MyEclipse新建web project和navicat110_mysql_en工具

    首先注意几点: 1.eclipse web项目:项目名称不得超过五个字符,要求全部小写,不管变量名.类名.函数名.文件名,在没有特殊理由的时候,不要用下划线,同时表名和类名用两个单词,尽量不要用Stu ...

  6. opencv算法学习

    1.改变图像的亮度和对比度: 算法介绍:对每一点像素值的r,g,b,值进行乘法和加法的运算. 代码使用: ; y < image.rows; y++ ) { ; x < image.col ...

  7. C#_基础,初始化器

    对象初始化器 在没有对象初始化器之前,我们创建一个对象大概需要经过这么两个步骤,首先new一个对象,然后给每个字段赋值.而有了对象初始化器之后,原本需要几行代码才能完成的任务变成一行代码就可以完成,简 ...

  8. ReactJS入门(二)—— 组件的生命周期

    如果你熟悉avalon,使用过 data-include-rendered 和 data-include-loaded 等回调方法,那么你会很好地理解React组件的各个生命周期. 说白了其实就是Re ...

  9. .NET实现微博粉丝服务平台接口

    [文章摘要]Senparc.Weixin.MP虽然是微信公众号的SDK,但由于易信公众号和新浪微博粉丝服务平台也提供了微信兼容接口,所以也可以使用其快速实现相应的服务,当然微博由于与微信存在差异,如果 ...

  10. Javascript本质第二篇:执行上下文

    在上一篇文章<Javascript本质第一篇:核心概念>中,对Javascript执行上下文做了解释,但是这些都是基于Javascript标准中对执行上下文的定义,也就是说理论上的东西,本 ...