###《More Effective C++》- 异常
More Effective C++
#@author: gr
#@date: 2015-05-24
#@email: forgerui@gmail.com
九、利用destructors避免泄漏资源
所谓RAII即"资源获取即是初始化的时候",所以就必须对资源进行释放。以一个对象存放资源,并依赖对象的析构函数释放资源。
把资源封装到对象体内,这样在发生异常时,对象析构时调用析构函数,释放资源,可以避免资源泄漏。
如果异常是在资源获取过程中抛出的,查看第十条;如果异常是在析构过程中发生的,查看第十一条。
十、在constructors内阻止资源泄漏
如果一个函数在constructor中发生了异常,它的析构函数不会被调用,因为这个对象的构造并没有完成。所以我们指望析构函数来帮我们清理。
BookEntry::BookEntry(const string& name, address, const string& imageFileName, const string& audioClipFileName)
:theName(name), theAddress(address), theImage(0), theAudioClip(0)
{
theImage = new Image(imageFileName);
theAudioClip = new AudioClip(audioClipFileName); //如果AudioClip对象构造出现异常,那么构造好的Image对象将无法释放
}
以智能指针取代"pointer class members",这样以局部对象管理资源,当发生异常,BookEntry自动销毁时,也不要手动删除它们所指的对象。
十一、禁止异常(exception)流出destructors之外
调用destructor有两种情况:
- 对象在正常状态下被销毁
- 对象被
exception处理机制销毁
我们讨论异常时的情况,在外面的catch块中调用析构函数,这样如果析构函数报错,还是会调用std::terminate()终止程序。避免这种情况,可以在catch块中再加一个try-catch,但这样做显得有些极端。方法是在可能产生异常的析构函数中进行异常的捕获,并且什么也不做。
Session::~Session()
{
try{
logDestruction(this);
}catch(...){}
}
十二、了解 "抛出一个exception" 与 "传递一个参数" 或 "调用一个虚函数" 之间的差异
catch子句的参数类型使用
pass by reference方法,可以对异常的修改起作用并且提高效率,一般将catch的参数类型定义为引用类型。查找匹配的
catch子句a. 类型转换支持的比函数参数匹配的少,不支持算术类型转换和类类型转换,只支持如下:
- 非常量向常量转型
- 派生类向基类转型
- 数组被转换成指向数组类型的指针
b. 匹配顺序
函数匹配是best-fit进行的,选择匹配参数最好的,而异常问题是按first fit进行,按照顺序进行,所以要把子类(及特殊类型)放在通用类型的前面。异常抛出
异常抛出:
Exception ex;
throw ex;
异常抛出时,不论catch是以什么方式捕获
by reference和by value都需要进行复制,到catch子句上的正是这个异常的副本。异常重新抛出:
catch (exception& ex)
{
//....
throw; //重新抛出此exception,使它继续传播
}
catch (exception& ex)
{
//....
throw ex; //传播被捕捉的exception的一个副本
}
十三、以by reference方式捕捉exceptions
同函数传递参数一样,为了效率和正确性,还是以引用传递比较好。
Catch exceptions by reference !
十四、明智运用exception specifications
所谓exception specifications就是在函数后面加上要抛出的异常类型。如下:
void f2() throw(int); //保证只抛出int类型的异常
void f3() throw(); //保证不抛出任何异常
但exception specifications也会带来问题,当抛出一个未列于其exception specification的exception,这个错误会在运行时检验出来,于是特殊函数unexpeted会被自动调用,终止程序。
void f1(); //f1并没有进行exception specification
void f2() throw(int)
{
//....
f1(); //调用f1(),可能抛出各种异常
//....
}
解决方法是如果被调用的函数没有exception specifications,那么调用函数也不要加exception specifications。
同时,在函数指针中调用函数时,也会进行这种检验,如实现一些注册的回调函数时。
第二个问题就是避免将exception specifications放在“需要类型自变量”的template身上。因为template必然以某种方式使用其类型参数,所以不应该template和exception specifications混合使用。
第三个问题是在处理“系统”可能抛出的exceptions。如在使用new operator时,可能抛出bad_alloc异常。
然而,如果你有一些函数都写了exception specifications,那么想要调用其它库中没有exception specifications的函数,我们只能主动将这些函数中抛出的其它异常,转换成UnexceptedException异常。
class UnexceptedException{};
void convertUnexcepted()
{
throw UnexceptedException();
}
set_unexpected(convertUnexpected);
但需要在exception specification中加入UnexceptedException,否则terminate还是被调用。
还有一种做法是将exceptions转换为一个标准类型bad_exception,它的实现依赖于:如果非预期函数的替代者重新抛出当前的(current)exception,它会被标准类型bad_exception代替。所以在exception specifications加入bad_exception即可:
void convertUnexcepted(){
throw; //重新抛出当前exception
}
set_unexpected(convertUnexpected);
exception specifications是一把双面刃,在“函数希望抛出什么样的exception”方面给出了卓越的说明,但同时在“违反exception specifications”时结局也比较悲惨。
十五、了解异常处理(exception handling)的成本
为了支持exception,程序必须做大量簿记工作。“天下没有免费的午餐”就是这个道理。必须在每一个try语句块的进入点和离开点做记号,针对每个try语句,必须记录对应的catch子句能处理的exceptions类型。
使用try语句块代码大约膨胀5%~10%, 执行速度也将下降。
如果有性能上的问题,利用分析工具(profiler)分析程序性能的瓶颈问题。
###《More Effective C++》- 异常的更多相关文章
- Java异常(二) 《Effective Java》中关于异常处理的几条建议
概要 本章是从<Effective Java>摘录整理出来的关于异常处理的几条建议.内容包括:第1条: 只针对不正常的情况才使用异常第2条: 对于可恢复的条件使用被检查的异常,对于程序错误 ...
- Effective C++ 条款08:别让异常逃离析构函数
1.别让异常逃离析构函数的原因 <Effective C++>第三版中条款08建议不要在析构函数中抛出异常,原因是C++异常机制不能同时处理两个或两个以上的异常.多个异常同时存在的情况下, ...
- 《More Effective C++ 》读书笔记(二)Exception 异常
这事篇读书笔记,只记录自己的理解和总结,一般情况不对其举例子具体说明,因为那正是书本身做的事情,我的笔记作为梳理和复习之用,划重点.我推荐学C++的人都好好读一遍Effective C++ 系列,真是 ...
- Effective java笔记(八),异常
57.只针对异常的情况才使用异常 try { int i = 0; while(true) range[i++].climb(); }catch(ArrayIndexOutOfBoundsExcept ...
- Effective Java 读书笔记之八 异常
一.只针对异常的情况才使用异常 1.类具有状态相关的方法时,可采用状态测试方法和可识别的返回值两个策略. 二.对可恢复的情况使用受检异常,对编程错误使用运行时异常 1.期望调用者能够适当恢复的情况,应 ...
- Effective C++ -----条款29:为“异常安全”而努力是值得的
异常安全函数(Exception-safe functions)即使发生异常也不会泄露资源或允许任何数据结构败坏.这样的函数区分为三种可能的保证:基本型.强烈型.不抛异常型. “强烈保证”往往能够以c ...
- Effective C++ -----条款08: 别让异常逃离析构函数
析构函数绝对不要吐出异常.如果一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉任何异常,然后吞下它们(不传播)或结束程序. 如果客户需要对某个操作函数运行期间抛出的异常作出反应,那么class应 ...
- [Effective Java]第九章 异常
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
- Effective C++ Item 29 为”异常安全”而努力是值得的
本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 经验:异常安全函数即使发生异常也不会泄漏资源或同意不论什么数据结构败坏.这种函数区分为三种 ...
随机推荐
- 基于redis的IP地址快速查询
在一些大数据处理中,我们需要用到IP地址查询,一般为了查询一个IP属于哪个地址,我们通常需要根据一个IP数据库来查询,网络上比较常用的IP库是纯真IP数据库.IP数据库里面的记录一般存储方式为IP的开 ...
- PT100运算放大器电路
运放输出电压<=运放电源电压,电源电压能决定它的最大输出能力即动态范围,若是电源为0-5v,则输出就只能在这之间. 其次要是放大电路,反馈必须接成负反馈 由于我这次使用的电源是5V,要是采用两 ...
- Keil µVision4 中出现中文乱码的解决办法
首先得说一下,以前都没有遇到过类似的问题,但是看到有个同学曾经满篇的乱码那叫个心疼. 这里我所说的办法其实只是格式转换的问题,对于其他原因造成的,可能会在以后遇到的时候再来处理了.另外,在将代码文件转 ...
- rxjs5.X系列 —— Combination/Multicasting系列 api 笔记
欢迎指导与讨论 :) 前言 本文是笔者翻译 RxJS 5.X 官网各类operation操作系列的的第三篇 -- Combination组合与Multicasting广播.如有错漏,希望大家指出提醒O ...
- 设计Account 对象如下: private long id; private double balance; private String password; 要求完善设计,使得该Account 对象能够自动分配id。 给定一个List 如下:
package homework005; public class Account { private long id; private double balance; private String ...
- JNI-使用RegisterNatives注册本地方法
转自: http://blog.chinaunix.net/uid-26009923-id-3410141.html 1. 以前在jni中写本地方法时,都会写成 Java_com_example_he ...
- 域名的MX设置及校验方法
国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...
- 【转】SQL语句中的正则表达示
正则表达式作用是匹配方本,将一个模式(正则表达式)与一个文本串进行比较. MySQL用WHERE子句对正则表达式提供了初步的支持,允许你指定用正则表达式过滤SELECT检索出的数据. MySQL仅支持 ...
- git 回退和删除操作
今天不小心把分支的commit提交到master上了.衰 主要通过下面几个命令解决了,很简单记录一下. git reset –hard 回退到某一个版本git push origin :xxxx ...
- 【翻译】Ext JS最新技巧——2014-5-12
原文:mkt_tok=3RkMMJWWfF9wsRoluazJZKXonjHpfsX77OQlXK%2B%2FlMI%2F0ER3fOvrPUfGjI4AT8NjI%2BSLDwEYGJlv6SgFS ...