Item 8:析构函数不要抛出异常 Effective C++笔记

Item 8: Prevent exceptions from leaving destructors.

析构函数不要抛出异常

因为析构函数经常被自己主动调用,在析构函数中抛出的异常往往会难以捕获,引发程序非正常退出或没有定义行为。 比如,对象数组被析构时。会抛出多于一个的异常,然而同一时候存在的异常在C++标准中是禁止的, 因此程序会非正常退出:

class Widget {
public:
  ~Widget() { ... }            // assume this might emit an exception
};
void doSomething(){
  std::vector<Widget> v;
}                              // v is automatically destroyed here

事实上,容器中的对象在析构时抛出异常还会引起兴许的对象无法被析构。导致资源泄漏。 这里的资源能够是内存,也能够是数据库连接。或者其它类型的计算机资源。

析构函数是由C++来调用的,源码中不包括对它的调用,因此它抛出的异常不可被捕获。 对于栈中的对象而言,在它离开作用域时会被析构。对于堆中的对象而言。在它被delte时析构。

请看:

class C{
public:
    ~C(){ throw 1;}
};
void main(){
    try{
        C c;
    }
    catch(int e){}
}

析构的异常并不会被捕获。由于try{}代码块中仅仅有一行代码C
c
。它并未抛出异常。 经Homebrew gcc 5.1.0编译后,执行时会产生这种错误输出:

libC++abi.dylib: terminating with uncaught exception of type int

或许你认为在try中用delete手动释放堆对象就能够捕获异常。我们来试试:

C *p = new C;
try{
    delete p;
}
catch(int e){}

上述代码会给出相同的错误输出:

libC++abi.dylib: terminating with uncaught exception of type int

这仅仅能说明delete并非对析构函数的直接调用,它仅仅是一个keyword。

析构函数还是由C++调用的。
其实。假设上面不delete的话,程序不会产生错误,此时p属于内存泄露。
这些内存是在程序退出后由操作系统来回收的。

那么在析构函数中。应处理掉可能的异常。保证对象可以被完整地释放。 由于析构函数中总会出现非安全的代码,我们仅仅能吞掉异常,或者退出程序。这样:

class DBConn{
public:
    ~DBConn{
        if(!closed){
            try{
                db.close();
            }
            catch(...){
              cerr<<"数据库关闭失败"<<endl;
              // 或者直接退出程序
                // std::abort();
            }
        }
    }
private:
    DBConnection db;
};

另外值得一提的是,上述catch(...)中的...并非省略号,它是合法标识符,表示不确定的形參。

可是对于一个完好的设计,我们须要让客户知道这里发生了异常。

在此仅仅需为不安全语句提供一个新的函数。在析构函数中我们还是运行默认操作(忽略、记录、或者结束程序)。

class DBConn{
public:
    void close(){
        db.close();
    }
    ...

这个常规方法给了客户自行关闭数据库并处理异常的机会。当然假设他放弃这个机会, 便不能怪罪于我们让程序退出或者吞掉异常了。



除非注明。本博客文章均为原创,转载请以链接形式标明本文地址: http://harttle.com/2015/07/26/effective-cpp-8.html

时间: 2024-10-08 09:29:54

Item 8:析构函数不要抛出异常 Effective C++笔记的相关文章

C++基础和STL,Effective C++笔记

C++基础 static static变量存储在静态数据区 相对于function:在函数内,变量,内存只被分配一次,多次调用值相同 相对于其他模块(.c文件):变量和函数,不能被模块外其他函数访问(private) 相对于类:类中的static变量和函数属于整个类,而不是对象 全局变量 VS 全局静态变量 若程序由一个源文件构成时,全局变量与全局静态变量没有区别. 若程序由多个源文件构成时,全局变量与全局静态变量不同:全局静态变量使得该变量成为定义该变量的源文件所独享,即:全局静态变量对组成该

Effective C++笔记04:设计与声明

条款18:让接口容易被正确使用,不易被误用 1,好的接口很容易被正确使用,不容易被误用.你应该在你的所有接口中努力达成这些性质. 2,"促进正使用"的办法包括接口的一致性,以及与内置类型的行为兼容. 3,"阻止误用"的办法包括建立新类型,限制类型上的操作,束缚对象值,以及消除客户的资源管理责任. 4,shared_ptr支持定制型删除器.这可以防范DLL问题,可以用来自动解除互斥锁. 条款19:设计class犹如设计type 博客地址:http://blog.csd

Effective C++笔记:构造/析构/赋值运算

条款05:了解C++默默编写并调用哪些函数 默认构造函数.拷贝构造函数.拷贝赋值函数.析构函数构成了一个类的脊梁,只有良好的处理这些函数的定义才能保证类的设计良好性. 当我们没有人为的定义上面的几个函数时,编译器会给我们构造默认的. 当成员变量里有const对象或引用类型时,编译器会不能合成默认的拷贝赋值函数:当一个基类把它的拷贝赋值函数定义为private时,它的派生类也不无生成默认的拷贝赋值函数,因为它无法完成基类成份的赋值. 条款06:若不想使用编译器自动生成的函数,就该明确拒绝 将拷贝构

Effective C++笔记06:继承与面向对象设计

关于OOP 博客地址:http://blog.csdn.net/cv_ronny 转载请注明出处! 1,继承可以是单一继承或多重继承,每一个继承连接可以是public.protected或private,也可以是virtual或non-virtual. 2,成员函数的各个选项:virtual或non-virtual或pure-virtual. 3,成员函数和其他语言特性的交互影响:缺省参数值与virtual函数有什么交互影响?继承如何影响C++的名称查找规则?设计选项有如些?如果class的行为

Effective c++(笔记)之继承关系与面向对象设计

1.公有继承(public inheritance) 意味着"是一种"(isa)的关系 解析:一定要深刻理解这句话的含义,不要认为这大家都知道,本来我也这样认为,当我看完这章后,就不这样认为了. 公有继承可以这样理解,如果令class D以public 的形式继承了class B ,那么可以这样认为,每一个类型为D的对象同时也可以认为是类型为B的对象,但反过来是不成立的,对象D是更特殊化更具体的的概念,而B是更一般化的概念,每一件事情只要能够施行于基类对象身上,就一定可以应用于派生类对

Effective c++(笔记) 之 类与函数的设计声明中常遇到的问题

1.当我们开始去敲代码的时候,想过这个问题么?怎么去设计一个类? 或者对于程序员来说,写代码真的就如同搬砖一样,每天都干的事情,但是我们是否曾想过,在c++的代码中怎么样去设计一个类?我觉得这个问题可比我们"搬砖"重要的多,大家说不是么? 这个答案在本博客中会细细道来,当我们设计一个类时,其实会出现很多问题,例如:我们是否应该在类中编写copy constructor 和assignment运算符(这个上篇博客中已说明),另外,我们是让编写的函数成为类的成员函数还是友元还是非成员函数,

Effective C++笔记05:实现

条款26:尽可能延后变量定义式的出现时间 博客地址:http://blog.csdn.net/cv_ronny 转载请注明出处! 有些对象,你可能过早的定义它,而在代码执行的过程中发生了导常,造成了开始定义的对象并没有被使用,而付出了构造成本来析构成本. 所以我们应该在定义对象时,尽可能的延后,甚至直到非得使用该变量前一刻为止,应该尝试延后这份定义直到能够给它初值实参为止. 这样做的好处是:不仅可以避免构造(析构)非必要对象,还可以避免无意义的default构造行为. 遇到循环怎么办?此时往往我

C++关于构造函数 和 析构函数 能否抛出异常的讨论

构造函数和析构函数分别管理对象的建立和释放,负责对象的诞生和死亡的过程.当一个对象诞生时,构造函数负责创建并初始化对象的内部环境,包括分配内存.创建内部对象和打开相关的外部资源,等等.而当对象死亡时,析构函数负责关闭资源.释放内部的对象和已分配的内存. 在对象生死攸关的地方,如果程序代码出现问题,常常会发生内存泄漏,从而产生可能危害系统运行的孤魂野鬼.大量的事实表明,业务逻辑代码写得非常严谨的程序在运行中仍然发现存在内存泄露,大都是构造和析构部分的代码存在问题. 而许多程序员都习惯于面向对象的编

C++构造函数、析构函数与抛出异常

[本文链接] http://www.cnblogs.com/hellogiser/p/constructor-destructor-exceptions.html [问题] 构造函数可以抛出异常么?析构函数可以吗? [分析] 从语法上来说,构造函数和析构函数都可以抛出异常.但从逻辑上和风险控制上,构造函数可以,析构函数不推荐抛出异常. (1)构造函数可以抛出异常 无论何时,从构造函数中抛出异常都是可以的.动态创建对象要进行两个操作:分配内存和调用构造函数.若在分配内存时出错,会抛出bad_all