1,问题:

1,如果在 main 函数中抛出异常会发生什么?

1,不处理,则崩溃;

2,如果异常不处理,最后会传到哪里?

3,下面的代码输出什么?

4,异常的最终处理编程实验:

 #include <iostream>

 using namespace std;

 class Test
{
public:
Test()
{
cout << "Test()";
cout << endl;
} ~Test()
{
cout << "~Test()";
cout << endl;
}
}; int main()
{
static Test t; // BCC Test();
// VC 2010 Test();
// G++ Test(); throw ; // BCC Abnormal program termination;
// VC 对话框弹出来;
// G++ terminate called after throwing an instance of 'int'; return ;
}

1,有了线索,main() 函数中抛出的异常也许会被一个最终的函数处理;

5,异常无法被处理的情况:

1,如果异常无法被处理,terminate() 结束函数会被自动调用;

2,默认情况下,terminate() 调用库函数 abort() 终止程序;

1,三款编译器在 terminate() 函数内部实现上面有点儿差异;

2,比如打印字符串提示当前程序出来异常,弹出对话框告诉用户当前应程序要异常终止(实际还没终止);

3,abort() 函数使得程序执行异常而立即退出;

4,C++ 支持替换默认的 terminate() 函数实现;

1,C++ 可以自定义结束函数,自定义最终处理异常的函数来替换预定义的 terminate();

6,terminate() 函数的替换:

1,自定义一个无返回值无参数的函数;

1,不能抛出任何异常;

1,最后处理异常的函数了,所以不能抛出任何异常;

2,必须以某种方式结束当前程序;

1,没有做到这一点儿,程序出现什么样的行为就不得而知了,当然 Windows 和 Linux 系统会将这个程序自动终止;

2,调用 set_terminate() 设置自定义的结束函数;

1,参数类型为 void(*)();

1,函数指针,没有参数、没有返回值;

2,返回值为默认的 terminate() 函数入口地址;

7,自定义结束函数编程实验:

 #include <iostream>
#include <cstdlib>
#include <exception> // C++ 标准库中与异常相关的头文件; using namespace std; void my_terminate()
{
cout << "void my_terminate()" << endl;
exit(); // 结束当前的程序;可以确保所有的全局对象和静态局部对象全部都正常析构;
// abort(); // “已放弃”是这个函数打印出来的,这个函数是异常终止一个程序,并且异常终止的时候不会调用任何对象的析构函数;
} class Test
{
public:
Test()
{
cout << "Test()";
cout << endl;
} ~Test()
{
cout << "~Test()";
cout << endl;
}
}; int main()
{
set_terminate(my_terminate); static Test t; throw ; return ;
}

1,打印结果:

1,exit(1) 的时候:

Test()

void my terminate()

~Test()

2,abort() 的时候:

Test()

void my terminate()

已放弃

2,在 main() 函数中扔出的异常如果没有被处理,会被最终的一个全局结束函数处理掉;

3,C++ 编译器之间存在差异;

8,问题:

1,如果析构函数中抛出异常会发生什么情况?

1,前面说有可能导致资源无法释放,内存泄漏等;

2,这里有两点:

1,析构函数是释放资源的地方,如果抛出异常,有可能导致资源无法正确的释放;

2,在析构函数中抛出异常有可能导致全局的结束函数 terminate() 被重复的调用,这是很可怕的,有可能让我们的系统进入一个不稳定的状态;

2,析构函数抛出异常编程实验:

 #include <iostream>
#include <cstdlib>
#include <exception> using namespace std; void my_terminate()
{
cout << "void my_terminate()" << endl;
// exit(1);
abort();
} class Test
{
public:
Test()
{
cout << "Test()";
cout << endl;
} ~Test()
{
cout << "~Test()";
cout << endl; throw ;
}
}; int main()
{
set_terminate(my_terminate); static Test t; throw ; return ;
}

1,打印结果:

1,Test()

void my terminate()  // main() 中 throw 1 抛出异常后第一次执行 my_terminate() 函数的结果;

~Test()  // 调用 exit(),执行析构函数;

void my terminate()  // 在执行析构函数的时候,再次扔出异常 throw 2,第二次执行 my_terminate() 函数;

2,my_terminate() 函数:

1,作为最后一个被调用的异常处理函数,任务是很重的;

2,要负责进行应用程序级别的资源释放;

3,第一次调用的时候,所有的资源已经被释放完了,第二次调用就类似对堆空间的内存第二次释放,造成应用程序的不稳定,因为这里 Linux 系统非常稳定,会帮我们处理好连续释放资源的问题;

4,如果进行嵌入式开发,操作系统就不见得有这样的能力对每一个系统做这样的监控,产生的行为使未定义的;

5,这也解释了 C++ 默认调用的是 abort() 而不是 exit(1),因为 abort() 直接的强制结束当前的应用程序,不会调用析构函数,就是拍析构函数中扔出异常;

9,小结:

1,如果异常没有被处理,最后 terminate() 结束整个程序;

2,terminate() 是整个程序释放系统资源的最后机会;

3,结束函数可以自定义,但不能继续抛出异常;

4,析构函数中不能抛出异常,可能导致 terminate() 多次调用;

C++异常处理的深入理解的更多相关文章

  1. Java异常处理机制 —— 深入理解与开发应用

    本文为原创博文,严禁转载,侵权必究! Java异常处理机制在日常开发中应用频繁,其最主要的不外乎几个关键字:try.catch.finally.throw.throws,以及各种各样的Exceptio ...

  2. 对try-catch-finally异常处理的最新理解

    try{ ...... }catch(......){ }finally{ ...... } 这个结构是用来处理Java所有可能出现的异常的,这个我很早其实就已经学过,不过最近看了个视频,感觉自己虽然 ...

  3. 谈谈你对Java异常处理机制的理解

    先谈谈我的理解:异常处理机制可以说是让我们编写的程序运行起来更加的健壮,无论是在程序调试.运行期间发生的异常情况的捕获,都提供的有效的补救动作,任何业务逻辑都会存在异常情况,这时只需要记录这些异常情况 ...

  4. 面向对象的异常处理之深入理解java异常处理机制

    什么是异常? 异常是对问题的描述,将问题的对象进行封装: 异常体系的特点:异常体系中的所有类以及建立的对象: 都具有可抛性,也就是说可以被throw和throws关键字所操作,只有异常体系具有该特点: ...

  5. 从C#到Objective-C,循序渐进学习苹果开发(4)--代码块(block)和错误异常处理的理解

    本随笔系列主要介绍从一个Windows平台从事C#开发到Mac平台苹果开发的一系列感想和体验历程,本系列文章是在起步阶段逐步积累的,希望带给大家更好,更真实的转换历程体验.本文继续上一篇随笔<从 ...

  6. aop原理及理解

    概念 Aspect Oriented Programming,面向切面编程,实际上它是一个规范.一种设计思路,总之是抽象的. 先上图 使用目的 从项目结构上来说 对业务逻辑的各个部分进行隔离,降低业务 ...

  7. 第6章 AOP与全局异常处理6.1-6.4 慕课网微信小程序开发学习笔记

    第6章 AOP与全局异常处理 https://coding.imooc.com/learn/list/97.html 目录: 第6章 AOP与全局异常处理6-1 正确理解异常处理流程 13:236-2 ...

  8. 第6章 AOP与全局异常处理6.5-6.11 慕课网微信小程序开发学习笔记

    https://coding.imooc.com/learn/list/97.html 目录: 第6章 AOP与全局异常处理6-1 正确理解异常处理流程 13:236-2 固有的处理异常的思维模式与流 ...

  9. 面向对象程序设计-C++ Stream & Template & Exception【第十五次上课笔记】

    这是本门<面向对象程序设计>课最后一次上课,刚好上完了这本<Thinking in C++> :) 这节课首先讲了流 Stream 的概念 平时我们主要用的是(1)在屏幕上输入 ...

随机推荐

  1. linux下创建svn仓库及用户

    1 Linux下创建svn仓库 1.1 启动SVN服务 svnserve -d -r  /SVNRootDirectry 其中SVNRootDirectry是你的SVN 根目录,例如192.85.1. ...

  2. scp - 安全复制(远程文件复制程序)

    总览 SYNOPSIS scp -words [-pqrvBC1246 ] [-F ssh_config ] [-S program ] [-P port ] [-c cipher ] [-i ide ...

  3. Kvm --01 虚拟化基础概念

    目录 1. 虚拟化基础概念 01. 什么是虚拟化? 02. 为什么要用虚拟化? 03. 虚拟化在企业中的应用场景? 04. 虚拟化软件介绍 05. Kvm介绍 2. 安装部署Kvm 3. Kvm虚拟机 ...

  4. CS184.1X 计算机图形学导论 HomeWork1

    最容易填写的函数就是left.输入为旋转的角度,当前的eye与up这两个三维向量 void Transform::left(float degrees, vec3& eye, vec3& ...

  5. ltp-ddt nand_ecc_tests

    NAND_S_FUNC_ECC_2K_BCH8_8ERRS_NO_OOB_ERR source "common.sh"; nandecc_tests.sh -r "0:0 ...

  6. Go的学习 append的使用

    1. package main; import "fmt" func test () { ],,,,}; s:=a[:] fmt.Printf(]) s=append(s,); s ...

  7. Python 无法安装PyAudio问题

    一.错误与原因 在Windows上没有用于Python 3.7的轮子(预构建包)(有一个用于Python 2.7和3.4到3.6),因此需要在PC上准备构建环境以使用此包.因为有些软件包很难在Wind ...

  8. JS中JSON.stringify()方法,将js对象(json串)转换成字符串,传入服务器

    JSON 通常用于与服务端交换数据. 在向服务器发送数据时一般是字符串. 我们可以使用 JSON.stringify() 方法将 JavaScript 对象转换为字符串. 语法 JSON.string ...

  9. Did you mean to run dotnet SDK commands

    把所有的net core的sdk,runtime,hosting通通卸载重新安装了2.1.1版本,发现再运行dotnet命令就是下面信息: C:\Users\Administrator>dotn ...

  10. sql判断中文、数字、英文

    IF OBJECT_ID('DBO.GET_NUMBER2') IS NOT NULL DROP FUNCTION DBO.GET_NUMBER2 GO )) ) AS BEGIN BEGIN ,'' ...