C++程序抛出异常后执行顺序

1 析构函数中是否可以抛出异常
  首先我们看一个常见的问题,析构函数中是否可以抛出异常。答案是C++标准指明析构函数不能、也不应该抛出异常
  C++异常处理模型是为C++语言量身设计的,更进一步的说,它实际上也是为C++语言中面向对象而服务的。C++异常处理模型最大的特点和优势就是对C++中的面向对象提供了最强大的无缝支持。那么如果对象在运行期间出现了异常,C++异常处理模型有责任清除那些由于出现异常所导致的已经失效了的对象(也即对象超出了它原来的作用域),并释放对象原来所分配的资源, 这就是调用这些对象的析构函数来完成释放资源的任务,所以从这个意义上说,析构函数已经变成了异常处理的一部分。
  下面我们来看看析构函数中不能抛出异常的两个理由:
  1)如果析构函数抛出异常,则异常点之后的程序不会执行,如果析构函数在异常点之后执行了某些必要的动作比如释放某些资源,则这些动作不会执行,会造成诸如资源泄漏的问题。
  2)通常异常发生时,c++的机制会调用已经构造对象的析构函数来释放资源,此时若析构函数本身也抛出异常,则前一个异常尚未处理,又有新的异常,会造成程序崩溃的问题。
  那么当无法保证在析构函数中不发生异常时, 该怎么办?
  其实还是有很好办法来解决的。那就是把异常完全封装在析构函数内部,决不让异常抛出函数之外。这是一种非常简单,也非常有效的方法。

//析构函数
~Class()
{
  try{
  }
  catch(){  //这里可以什么都不做,只是保证catch块的程序抛出的异常不会被扔出析构函数之外。
   }
}

2 程序抛出异常后会怎样
  下面我们通过一个程序来观察当程序中抛出异常了是否会调用析构函数,异常抛出中throw()后面的语句是否还会执行。程序如下,我们创建一个类,然后构造一个类对象,当抛出异常我们看程序是否会进入析构函数以及throw()抛出异常后面的程序:

#include<iostream>
using namespace std;

class setTry{
public:
    setTry(){  //构造函数
        cout << "start!" << endl; // 1
    }
    ~setTry(){ //析构函数
        cout << "end!" << endl;   // 4
    }
    void dosomething(){
        cout << "do something!" << endl; //类方法
    }
};
int main(void)
{
    setTry newOne;
    try{
        throw("error!");  //直接抛出异常  
        newOne.dosomething();
    }
    catch (char* one){    //接收char*类异常  
        cout << one << endl;   // 2
    }
    catch (...){          //接收其他类型异常
        cout << "..." << endl;
    }
    cout << "return 0!"<<endl; // 3
    return 0;
}

  上面程序运行结就是按标注的1、2、3、4步骤输出的,结果如下图所示:

这里写图片描述

  从运行结果就可以看出,抛出异常try内部的throw()后面程序不会再执行,而try外部后面的程序会继续执行。另外,析构函数在生存期结束也会被调用。


  参考文献:
  http://blog.csdn.net/u012398613/article/details/17469581

  • 4
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
C++中,try-catch语句用于捕获和处理异常。当程序中的代码可能会引发异常时,我们可以使用try块来包裹这部分代码,并使用catch块来捕获并处理异常。 try块中包含可能引发异常的代码,如果在try块中的代码引发了异常,那么程序会立即跳转到与之匹配的catch块中进行异常处理。 catch块用于捕获和处理异常。catch块可以指定捕获特定类型的异常,也可以使用通用的catch块来捕获所有类型的异常。在catch块中,我们可以编写处理异常的代码逻辑,比如输出错误信息、进行日志记录、进行恢复操作等。 下面是一个简单的示例代码,演示了try-catch语句的使用: ```cpp try { // 可能引发异常的代码 throw MyException(); // 抛出自定义异常 } catch (const MyException& e) { // 处理自定义异常 std::cout << "Caught MyException: " << e.what() << std::endl; } catch (const std::exception& e) { // 处理其他类型的异常 std::cout << "Caught std::exception: " << e.what() << std::endl; } catch (...) { // 处理未知类型的异常 std::cout << "Caught unknown exception" << std::endl; } ``` 在上面的代码中,try块中抛出了一个自定义异常MyException。catch块按照顺序进行匹配,首先匹配到与MyException类型匹配的catch块,然后执行其中的代码。如果没有匹配到特定类型的catch块,则会匹配到通用的catch块,执行其中的代码。如果连通用的catch块也没有匹配到,则异常会继续向上层传递,直到找到匹配的catch块或者程序终止。 需要注意的是,catch块中的参数可以是异常类型的引用或指针,通过该参数可以访问到抛出的异常对象。在上面的示例中,我们使用了const引用来捕获异常对象,并通过e.what()方法获取异常信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值