温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

异常处理深度解析(五十八)

发布时间:2020-08-05 19:01:12 来源:网络 阅读:480 作者:上帝之子521 栏目:编程语言

        我们之前学习了异常有关的知识,那么如果在 main 函数中中抛出异常会发生什么呢?如果异常不进行处理,最后会传到哪里呢?如下

异常处理深度解析(五十八)

        下来我们就来做个实验,代码如下

#include <iostream>

using namespace std;

class Test
{
public:
    Test()
    {
        cout << "Test()" << endl;
    }
    
    ~Test()
    {
        cout << "~Test()" << endl;
    }
};

int main()
{
    static Test t;
    
    throw 1;
    
    return 0;
}

        我们先来看看 g++ 编译器是怎样处理的

异常处理深度解析(五十八)

        我们看到在打印了构造函数的语句之后,下面还输出两句话,那么我们并没有在程序中定义这样的输出啊,这个到底是谁打印出来的呢?我们来看看 BCC 编译器

异常处理深度解析(五十八)

        在 BCC 编译器中输出了下面的一句话,也并不是我们定义的。我们再来看看 vs2010 编译器

异常处理深度解析(五十八)

        我们看到在 vs2010 编译器中弹出了一个对话框,我们并没有编写相关的代码。我们来看看编译器在背后究竟做了哪些事,如果异常无法被处理,terminate() 结束函数会被自动调用。默认情况下,terminate() 调用库函数 abort() 终止程序,abort() 函数使得程序执行异常而立即退出,C++ 支持替换默认的 terminate() 函数实现。

        terminate() 函数的替换:a> 它是自定义一个无返回值无参数的函数,不能抛出任何异常,必须以某种方式结束当前程序;b> 调用 set_terminate() 设置自定义的结束函数,参数类型为 void(*)(),返回值为默认的 terminate() 函数入口地址。

        下来我们来自定义 terminate() 函数。

#include <iostream>
#include <cstdlib>
#include <exception>

using namespace std;

void my_terminate()
{
    cout << "void my_terminate()" << endl;
    exit(1);
}

class Test
{
public:
    Test()
    {
        cout << "Test()" << endl;
    }
    
    ~Test()
    {
        cout << "~Test()" << endl;
    }
};

int main()
{
    set_terminate(my_terminate);
    
    static Test t;
    
    throw 1;
    
    return 0;
}

        我们再来看看编译结果,先在 g++ 编译器下

异常处理深度解析(五十八)

        我们看到程序正常运行结束了。我们来分析下,我们在 main 函数中调用 set_terminate() 设置结束函数 my_terminate()。在 main 函数中抛出了一个异常,被结束函数 my_terminate() 捕获到了,然后执行它里面的打印语句,进而执行到 exit(1) 正常退出。所以在最后退出的时候会去执行析构函数。我们再来看看 BCC 编译器呢

异常处理深度解析(五十八)

        我们看到和 g++ 编译器中的行为是一样的,再来看看 vs2010 编译器看看

异常处理深度解析(五十八)

        那么如果在析构函数中抛出异常会发生什么情况呢?我们来试试,在上面的程序中的析构函数抛出一个异常,看看编译结果

异常处理深度解析(五十八)

        我们看到最后又去调用内置的 Aborted 函数了。再来看看 BCC 编译器呢

异常处理深度解析(五十八)

        同样也是这样的情况,看看 vs2010 编译器

异常处理深度解析(五十八)

        我们来分析下,它先是在 main 函数中抛出异常,然后执行到 my_terminate() 函数中,进而退出(清理一切资源等),在退出的时候又去执行析构函数,又再次抛出异常,等于又要再次进行资源的释放,造成二次释放了。类似于在操作指针时,二次释放,所带来的后果是无法确定的。因此它最后会去调用库函数中的 abort() 函数。因此,我们在 main 函数中尽量不要抛出异常。通过对异常的深度学习,总结如下:1、如果异常没有被处理,最后 terminate() 结束整个程序;2、terminate() 是整个程序释放系统资源的最后机会;3、结束函数可以自定义,但不能继续抛出异常;4、析构函数中不能抛出异常,可能导致 terminate() 多次调用。


        欢迎大家一起来学习 C++ 语言,可以加我QQ:243343083

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI