异常处理深度解析(五十八)-创新互联

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

创新互联公司-专业网站定制、快速模板网站建设、高性价比汇川网站开发、企业建站全套包干低至880元,成熟完善的模板库,直接使用。一站式汇川网站制作公司更省心,省钱,快速模板网站建设找我们,业务覆盖汇川地区。费用合理售后完善,十载实体公司更值得信赖。

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

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

#include 

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 
#include 
#include 

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。

另外有需要云服务器可以了解下创新互联scvps.cn,海内外云服务器15元起步,三天无理由+7*72小时售后在线,公司持有idc许可证,提供“云服务器、裸金属服务器、高防服务器、香港服务器、美国服务器、虚拟主机、免备案服务器”等云主机租用服务以及企业上云的综合解决方案,具有“安全稳定、简单易用、服务可用性高、性价比高”等特点与优势,专为企业上云打造定制,能够满足用户丰富、多元化的应用场景需求。


网页名称:异常处理深度解析(五十八)-创新互联
转载来源:http://csdahua.cn/article/cocjjd.html
扫二维码与项目经理沟通

我们在微信上24小时期待你的声音

解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流