Effective C++ —— 别让异常逃离析构函数
2015-09-20 13:08
309 查看
C++ 并不禁止析构函数吐出异常,但不鼓励这么做。
1.如果一个被析构函数调用的函数可能抛出异常,则析构函数应该捕获这些异常,然后终止程序或吞下它们。
2.如果客户需要对某个函数在运行中抛出的异常做出反应,那么这个类应该提供一个普通函数(非析构函数)执行该操作。
v在销毁时有责任销毁容器内的所有Widget 对象,如果在销毁第一个对象时抛出了异常,剩下的对象还是应该被销毁。如果第二个对象被销毁时又出现了异常,这时对于C++而言异常有些多了,程序不是结束执行就是导致不明确行为。
使用标准库容器或数组等都有可能出现这样的情况,其实只要析构函数吐出异常,程序都会出现这样的问题。
第二个例子:DBConnection 类负责数据库连接
class DBConnection
{
public:
static DBConnection Create();
private:
void close();//关闭联机,释放资源,失败则抛出异常
};
为确保客户不忘记调用close()方法来释放资源,可以设计一个数据库资源管理类,在其析构函数中调用close().
class DBResManager
{
public:
DBResManager();
~DBResManager()
{
db.close();
}
private:
DBConnection db;
};
close()调用成功,一切都美好,一旦异常抛出,则DBResManager析构函数会传播异常。两个解决方案:
一.调用abort()终止程序
二.吞下异常
最好的方案是:重新设计DBResManager类的接口,让客户有机会处理异常
说明:当某个操作在失败时可能抛出异常,又必须对这个异常进行处理时,那么这个处理函数必须是非析构函数。因为在析构函数中抛出异常是个很糟糕的想法,结果就是过早地结束程序或发生不明确的行为。
1.如果一个被析构函数调用的函数可能抛出异常,则析构函数应该捕获这些异常,然后终止程序或吞下它们。
2.如果客户需要对某个函数在运行中抛出的异常做出反应,那么这个类应该提供一个普通函数(非析构函数)执行该操作。
class Widget { public: Widget(); ~Widget() {} }; void work() { std::vector<Widget> v; }
v在销毁时有责任销毁容器内的所有Widget 对象,如果在销毁第一个对象时抛出了异常,剩下的对象还是应该被销毁。如果第二个对象被销毁时又出现了异常,这时对于C++而言异常有些多了,程序不是结束执行就是导致不明确行为。
使用标准库容器或数组等都有可能出现这样的情况,其实只要析构函数吐出异常,程序都会出现这样的问题。
第二个例子:DBConnection 类负责数据库连接
class DBConnection
{
public:
static DBConnection Create();
private:
void close();//关闭联机,释放资源,失败则抛出异常
};
为确保客户不忘记调用close()方法来释放资源,可以设计一个数据库资源管理类,在其析构函数中调用close().
class DBResManager
{
public:
DBResManager();
~DBResManager()
{
db.close();
}
private:
DBConnection db;
};
close()调用成功,一切都美好,一旦异常抛出,则DBResManager析构函数会传播异常。两个解决方案:
一.调用abort()终止程序
DBResManager::~DBResManager() { try { db.close(); } catch() { std::abort();//终止程序,避免异常传播 } }
二.吞下异常
DBResManager::~DBResManager() { try { db.close(); } catch() { //吞下异常,程序继续执行,保证程序可以继续可靠执行 } }
最好的方案是:重新设计DBResManager类的接口,让客户有机会处理异常
class DBResManager { public: DBResManager(); void close() { db.close(); closed = true; } ~DBResManager() { if(!closed) { try { db.close(); } catch() { //吞下异常,程序继续执行,保证程序可以继续可靠执行 } } private: DBConnection db; bool closed; };
说明:当某个操作在失败时可能抛出异常,又必须对这个异常进行处理时,那么这个处理函数必须是非析构函数。因为在析构函数中抛出异常是个很糟糕的想法,结果就是过早地结束程序或发生不明确的行为。
相关文章推荐
- 漫谈c++11 Thread库之使写多线程程序
- c++中的时间戳
- c中const和c++中const的学习总结
- c++中的各种容器
- C++字符串string类常用操作详解(一)【初始化、遍历、连接】
- c++反汇编学习笔记-------------篇外1、if和switch效率的实测
- C语言指针数组与数组指针
- C++ trivial和non-trivial构造函数及POD类型
- Effective C++——条款47(第7章)
- C# C++
- c++反汇编学习笔记----------3.流程控制
- C++动态分配内存(new)和撤销内存(delete)
- C/C++中二维数组和指针关系分析
- 第3周项目3 求集合并集
- 理解C++ dynamic_cast
- C++运算符重载
- C++继承和多态
- c++的override关键字什么意思
- C/C++里面const的用法
- C++中类的构造函数调用顺序