zoukankan      html  css  js  c++  java
  • C++异常注意事项

    C++里面catch对于类型转换,限制比参数传递时候要多:

    不可以进行标准算术转换类的自定义转换:在函数参数匹配的过程中,可以进行很多的类型转换。但是在异常匹配的过程中,转换的规则要严厉。

    标准算术转换,指的是 short转成int 等等。异常catch的时候,不允许转换,指的是匹配的时候,就不会匹配上。比如下面:

    #include <iostream>
    #include <exception>
    #include <stack>
    
    using namespace std;
    
    int main() {
        std::cout << "Hello, World!" << std::endl;
    
        stack<int> stk;
        //stk.push(5);
        try {
            //stk.pop();
            short s = 5;
            throw s;
        }
        /*catch(runtime_error exception1) {
            std::cout << exception1.what() << endl;
        }*/
        catch(int &x) {
            std::cout << x << endl;
        }
        catch (...) {
            cout << "here catch" << endl;
        }
    
        return 0;
    }
    View Code

    输出:

    here catch
    View Code

    意味着,int对于short的catch没有接住。

    另外,异常处理机制的匹配过程是寻找最先匹配(first fit),函数调用的过程是寻找最佳匹配(best fit)。

    拷贝代价

    如果throw中抛出一个对象,那么无论是catch中使用什么接收(基类对象、引用、指针或者子类对象、引用、指针),在传递到catch之前,编译器都会另外构造一个对象的副本。也就是说,如果你以一个throw语句中抛出一个对象类型,在catch处通过也是通过一个对象接收,那么该对象经历了两次复制,即调用了两次复制构造函数。

    一次是在throw时,将“抛出到对象”复制到一个“临时对象”(这一步是必须的),然后是因为catch处使用对象接收,那么需要再从“临时对象”复制到“catch的形参变量”中; 如果你在catch中使用“引用”来接收参数,那么不需要第二次复制,即形参的引用指向临时变量。 

    析构、构造函数

     2. 析构函数应该从不抛出异常。如果析构函数中需要执行可能会抛出异常的代码,那么就应该在析构函数内部将这个异常进行处理,而不是将异常抛出去。

                         原因:在为某个异常进行栈展开时,析构函数如果又抛出自己的未经处理另一个异常,将会导致调用标准库 terminate 函数。而默认的terminate 函数将调用 abort 函数,强制从整个程序非正常退出。

                   3. 构造函数中可以抛出异常。但是要注意到:如果构造函数因为异常而退出,那么该类的析构函数就得不到执行。所以要手动销毁在异常抛出前已经构造的部分。

    具体可以看这一篇的说明:

    http://www.cnblogs.com/zhyg6516/archive/2011/03/08/1977007.html

    Linux的进程布局可以看:

    http://www.cnblogs.com/charlesblc/p/6415916.html

    具体讲,析构函数不能抛出异常,是因为只有析构函数抛出异常没问题,但是如果有其他函数抛出异常,并且调用了析构函数抛出异常的处理,那么会造成SEH的异常处理链的混乱。

    构造函数可以抛出异常,但是不建议。

    需要处理掉已经声明的内存,或者用RAII.

    对于多线程,异常是安全的

    对于每一个线程都有自己的 Thread Info/Environment Block. 维护自己的SEH结构。

  • 相关阅读:
    Easy | LeetCode 108. 将有序数组转换为二叉搜索树
    Medium | LeetCode 105 | 剑指 Offer 07. 从前序与中序遍历序列构造二叉树
    Easy | LeetCode 543. 二叉树的直径
    Easy | LeetCode 235 | 剑指 Offer 68
    Easy | LeetCode 236 | 剑指 Offer 68
    Medium | LeetCode 114. 二叉树展开为链表 | 先序遍历 | 递归 | 迭代
    Medium | LeetCode 538,1038. 把二叉搜索树转换为累加树
    Medium | LeetCode 230. 二叉搜索树中第K小的元素
    Easy | 剑指 Offer 54. 二叉搜索树的第k大节点
    stl(5)vector容器
  • 原文地址:https://www.cnblogs.com/charlesblc/p/6415451.html
Copyright © 2011-2022 走看看