成C++应用程序世界------异常处理

一、 概述

C++自身有着很强的纠错能力,发展到现在,已经建立了比較完好的异常处理机制。C++的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句,函数,结构和类,致使编译程序无法进行。还有一种是执行时发生的错误,一般与算法有关。

关于语法错误,不必多说,写代码时心细一点就能够解决。C++编译器的报错机制能够让我们轻松地解决这些错误。

第二种是执行时的错误,常见的有文件打开失败、数组下标溢出、系统内存不足等等。

而一旦出现这些问题,引发算法失效、程序执行时无故停止等故障也是常有的。这就要求我们在设计软件算法时要全面。比方针对文件打开失败的情况。保护的方法有非常多种,最简单的就是使用“return”命令。告诉上层调用者函数执行失败;第二种处理策略就是利用c++的异常机制,抛出异常。

二、c++异常处理机制

C++异常处理机制是一个用来有效地处理执行错误的很强大且灵活的工具。它提供了很多其它的弹性、安全性和稳固性,克服了传统方法所带来的问题.

异常的抛出和处理主要使用了下面三个keyword: try、 throw 、 catch 。

抛出异常即检測是否产生异常。在C++中。其採用throw语句来实现,假设检測到产生异常,则抛出异常。该语句的格式为:

throw 表达式;

假设在try语句块的程序段中(包含在当中调用的函数)发现了异常,且抛弃了该异常,则这个异常就能够被try语句块后的某个catch语句所捕获并处理,捕获和处理的条件是被抛弃的异常的类型与catch语句的异常类型相匹配。因为C++使用数据类型来区分不同的异常,因此在推断异常时。throw语句中的表达式的值就没有实际意义。而表达式的类型就特别重要。

try-catch语句形式例如以下 :

  1. try
  2. {
  3. 包括可能抛出异常的语句;
  4. }
  5. catch(类型名 [形參名]) // 捕获特定类型的异常
  6. {
  7. }
  8. catch(类型名 [形參名]) // 捕获特定类型的异常
  9. {
  10. }
  11. catch(...)    // 三个点则表示捕获全部类型的异常
  12. {
  13. }

【范例1】处理除数为0的异常。该范例将上述除数为0的异常能够用try/catch语句来捕获异常。并使用throw语句来抛出异常,从而实现异常处理,实现代码如代码清单1-1所看到的。

// 代码清单1-1

  1. #include<iostream.h>     //包括头文件
  2. #include<stdlib.h>
  3. double fuc(double x, double y) //定义函数
  4. {
  5. if(y==0)
  6. {
  7. throw y;     //除数为0,抛出异常
  8. }
  9. return x/y;     //否则返回两个数的商
  10. }
  11. void main()
  12. {
  13. double res;
  14. try  //定义异常
  15. {
  16. res=fuc(2,3);
  17. cout<<"The result of x/y is : "<<res<<endl;
  18. res=fuc(4,0); 出现异常,函数内部会抛出异常
  19. }
  20. catch(double)             //捕获并处理异常
  21. {
  22. cerr<<"error of dividing zero.\n";
  23. exit(1);                //异常退出程序
  24. }
  25. }

【范例2】自己定义异常类型 (在本文開始的代码中已经给出示范)

三、异常的接口声明

为了加强程序的可读性,使函数的用户可以方便地知道所使用的函数会抛出哪些异常,可以在函数的声明中列出这个函数可能抛出的全部异常类型。比如:

void fun() throw( A,B,C,D);

这表明函数fun()可能而且仅仅可能抛出类型(A,B,C,D)及其子类型的异常。

假设在函数的声明中没有包含异常的接口声明。则此函数能够抛出不论什么类型的异常,比如:

void fun();

一个不会抛出不论什么类型异常的函数能够进行例如以下形式的声明:

void fun() thow();

五、异常处理中须要注意的问题

1. 假设抛出的异常一直没有函数捕获(catch)。则会一直上传到c++执行系统那里,导致整个程序的终止

2. 一般在异常抛出后资源能够正常被释放,但注意假设在类的构造函数中抛出异常,系统是不会调用它的析构函数的,处理方法是:假设在构造函数中要抛出异常,则在抛出前要记得删除申请的资源。

3. 异常处理只通过类型而不是通过值来匹配的,所以catch块的參数能够没有參数名称。只须要參数类型。

4. 函数原型中的异常说明要与实现中的异常说明一致。否则easy引起异常冲突。

5. 应该在throw语句后写上异常对象时,throw先通过Copy构造函数构造一个新对象,再把该新对象传递给 catch.

那么当异常抛出后新对象怎样释放?

异常处理机制保证:异常抛出的新对象并不是创建在函数栈上,而是创建在专用的异常栈上。因此它才干够跨接多个函数而传递到上层,否则在栈清空的过程中就会被销毁。全部从try到throw语句之间构造起来的对象的析构函数将被自己主动调用。

但假设一直上溯到main函数后还没有找到匹配的catch块,那么系统调用terminate()终止整个程序,这样的情况下不能保证全部局部对象会被正确地销毁。

6. catch块的參数推荐採用地址传递而不是值传递,不仅能够提高效率,还能够利用对象的多态性。另外,派生类的异常扑获要放到父类异常扑获的前面,否则。派生类的异常无法被扑获。

7. 编写异常说明时。要确保派生类成员函数的异常说明和基类成员函数的异常说明一致,即派生类改写的虚函数的异常说明至少要和相应的基类虚函数的异常说明同样,甚至更加严格,更特殊。

#include "stdafx.h"
#include<stdlib.h>
#include<crtdbg.h>
#include <iostream>
// 内存泄露检測机制
#define _CRTDBG_MAP_ALLOC
#ifdef _DEBUG
#define new new(_NORMAL_BLOCK, __FILE__, __LINE__)
#endif  

// 自己定义异常类
class MyExcepction
{
public:  

        // 构造函数,參数为错误代码
        MyExcepction(int errorId)
        {
            // 输出构造函数被调用信息
            std::cout << "MyExcepction is called" << std::endl;
            m_errorId = errorId;
        }  

        // 拷贝构造函数
        MyExcepction( MyExcepction& myExp)
        {
            // 输出拷贝构造函数被调用信息
            std::cout << "copy construct is called" << std::endl;
            this->m_errorId = myExp.m_errorId;
        }  

       ~MyExcepction()
        {
            // 输出析构函数被调用信息
            std::cout << "~MyExcepction is called" << std::endl;
        }  

       // 获取错误码
        int getErrorId()
        {
            return m_errorId;
        }  

private:
        // 错误码
        int m_errorId;
};  

int main(int argc, char* argv[])
{
        // 内存泄露检測机制
        _CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF );  

        // 能够改变错误码,以便抛出不同的异常进行測试
        int throwErrorCode = 110;  

       std::cout << " input test code :" << std::endl;
       std::cin >> throwErrorCode;  

       try
       {
            if ( throwErrorCode == 110)
            {
                MyExcepction myStru(110);  

                // 抛出对象的地址 -> 由catch( MyExcepction*    pMyExcepction) 捕获
                // 这里该对象的地址抛出给catch语句,不会调用对象的拷贝构造函数
                // 传地址是提倡的做法,不会频繁地调用该对象的构造函数或拷贝构造函数
                // catch语句运行结束后,myStru会被析构掉
                throw    &myStru;
            }
            else if ( throwErrorCode == 119 )
            {
                MyExcepction myStru(119);  

                // 抛出对象,这里会通过拷贝构造函数创建一个暂时的对象传出给catch
                // 由catch( MyExcepction    myExcepction) 捕获
                // 在catch语句中会再次调用通过拷贝构造函数创建暂时对象复制这里传过去的对象
                // throw结束后myStru会被析构掉
                throw    myStru;
             }
             else if ( throwErrorCode == 120 )
             {
                  // 不提倡这种抛出方法
                  // 这样做的话,假设catch( MyExcepction*    pMyExcepction)中不运行delete操作则会发生内存泄露  

                  // 由catch( MyExcepction*    pMyExcepction) 捕获
                  MyExcepction * pMyStru = new MyExcepction(120);
                  throw pMyStru;
             }
             else
             {
                  // 直接创建新对象抛出
                  // 相当于创建了暂时的对象传递给了catch语句
                  // 由catch接收时通过拷贝构造函数再次创建暂时对象接收传递过去的对象
                  // throw结束后两次创建的暂时对象会被析构掉
                   throw MyExcepction(throwErrorCode);
             }
        }
        catch( MyExcepction*    pMyExcepction)
        {
             // 输出本语句被运行信息
               std::cout << "运行了 catch( MyExcepction*    pMyExcepction) " << std::endl;  

             // 输出错误信息
               std::cout << "error Code : " << pMyExcepction->getErrorId()<< std::endl;  

            // 异常抛出的新对象并不是创建在函数栈上,而是创建在专用的异常栈上,不须要进行delete
            //delete pMyExcepction;
        }
        catch ( MyExcepction myExcepction)
        {
            // 输出本语句被运行信息
            std::cout << "运行了 catch ( MyExcepction myExcepction) " << std::endl;  

            // 输出错误信息
            std::cout << "error Code : " << myExcepction.getErrorId()<< std::endl;
        }
        catch(...)
        {
             // 输出本语句被运行信息
             std::cout << "运行了 catch(...) " << std::endl;  

             // 处理不了,又一次抛出给上级
             throw ;
        }  

        // 暂停
        int temp;
        std::cin >> temp;  

       return 0;  
/*File : exception.cpp
 *Auth : sjin
 *Date : 20140515
 *Mail : [email protected]
 */

#include <iostream>
#include <exception>

using namespace std;

class MyExcception:public exception{

public:
	const char * what() const throw(){
		return "this is a exception";
	}
};

/*假设抛出异常,没有接收。程序将终止*/
void exception_1()throw(double,int,const char *,MyExcception)
{
	int i;
	cout << "输入1-3整数" <<endl;
	cin >> i;
	if(i == 1)throw MyExcception();
	if(i == 2)throw "hello";
	if(i== 3)throw 123;

	cout << "==========end =========" <<endl;

}
int main()
{
	try{

		exception_1();
	}catch(const char *ex){
		cout <<" exceptiong is occur!" <<ex<< endl;
	}catch(double e){
		cout << e << endl;
	}catch (int e){
		cout << e <<endl;
	}catch (MyExcception e){
		cout << "MyExcception is occur!" << e.what()<< endl;
	}

	return 0;
}

版权声明:本文博客原创文章,博客,未经同意,不得转载。

时间: 2024-08-13 03:16:06

成C++应用程序世界------异常处理的相关文章

走进C++程序世界------异常处理

一. 概述 C++自身有着非常强的纠错能力,发展到如今,已经建立了比较完善的异常处理机制.C++的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句,函数,结构和类,致使编译程序无法进行.另一种是运行时发生的错误,一般与算法有关. 关于语法错误,不必多说,写代码时心细一点就可以解决.C++编译器的报错机制可以让我们轻松地解决这些错误. 第二种是运行时的错误,常见的有文件打开失败.数组下标溢出.系统内存不足等等.而一旦出现这些问题,引发算法失效.程序运行时无故停止等故障也是常有的.这就要求

002 - 在安卓手机上学习C语言-Linux入门 通往程序世界之门-操作系统

Linux入门  通往程序世界之门--操作系统 在上一章中 , 我们讨论了为何要搭建编译环境, 那么多的上仙出场, 我相信大家还能记住的搭建编译环境的原因的. 在讨论的时候, 不知不觉地把Linux操作系统给提出来了, 其实, 使用Windows去教学可能会更方便一点. 毕竟大家都用熟了嘛 , 不过我们是在手机上编程, 手机上使用不了Windows上的工具, 所以Windows暂时是用不上了, 只好转入Linux系统的怀抱了. 在这一章中, 我会简单地介绍一下在Linux的下使用到的命令. 最后

我程序世界的“术”与“道”

17年前高考的时候,本人就立志要从事所谓的科技行业,所以在填写志愿的时候填写的是通信工程,顺便后面同意了服从志愿调剂.就因为服从调剂,我这个完全能上得了通信工程专业分数的人得到了"爱因斯坦"老先生的眷顾被调到物理专业了(在这里不得不吐槽一下高考招生的黑暗与混乱).本着对他老先生的好心就将就成为他的徒子徒孙吧.但经过一段时间,我才发现本人对物理一点兴趣没有,他老人家肯定对我是恨铁不成钢,彻底把我抛弃了.浑浑噩噩的读完一个学期后,在一次和同学的调侃中调侃到计算机程序设计.心中一种莫名的冲动

在安卓手机上学习C语言 - 安卓手机C/C++编译环境的搭建 : 程序世界的创建

   在安卓手机上学习C语言           安卓手机C/C++编译环境的搭建 : 程序世界的创建 在电脑上运行的QQ,手机上的QQ都是程序, 这些通电就能用的神奇玩意, 如果我说它们都是程序员用一个一个英文字母,数字,奇奇怪怪的符号创造出来的,我想那些没有任何概念的朋友可能会感到惊讶. 是的, 在没有揭开程序世界的神秘面纱之前, 一切都是那么神奇. 实际上, 程序确实是用一个一个英文字母(或者说拼音字母...)来写出来的, 就和写小说一样写出来的. 但是并不是写完之后把写出来的内容保存到一

这些基础知识你都了解吗?——《松本行弘的程序世界》读书笔记(上)

1. 前言 半个月之前买了这本书,还是经园子里的一位网友推荐的.到现在看了一半多,基础的都看完了,剩下的几章可做高级部分来看.这本书看到现在,可以说感触很深,必须做一次读书笔记! 关于这本书,不了解的可以去网上查查.作者是Ruby语言的创始人,可谓是程序世界中的高手,开卷有益,不管你是哪个层次的编程人员,相信都能或多或少的汲取到你想要的营养. 下面将总结一下看完本书我记录下的一些知识点.有的是书中的原话,有的是我个人的理解,供参考. 2. 面向对象 2.1 多态性 面向对象三大原则:继承.封装和

《松本行弘的程序世界》读书笔记

1. 前言 半个月之前买了这本书,还是经园子里的一位网友推荐的.到现在看了一半多,基础的都看完了,剩下的几章可做高级部分来看.这本书看到现在,可以说感触很深,必须做一次读书笔记! 关于这本书,不了解的可以去网上查查.作者是Ruby语言的创始人,可谓是程序世界中的高手,开卷有益,不管你是哪个层次的编程人员,相信都能或多或少的汲取到你想要的营养. 下面将总结一下看完本书我记录下的一些知识点.有的是书中的原话,有的是我个人的理解,供参考. 2. 面向对象 2.1 多态性 面向对象三大原则:继承.封装和

走进C++程序世界------IO标准库介绍

流概述 流是C++标准的组成部分,流的主要目标是,将从磁盘读取文件或将输入写入控制台屏幕的问题封装起来,创建流后程序员就可以使用它,流将负责处理所有的细节. IO类库 在C++输入?输出操作是通过C++系统提供的完成I/O操作的一组类实现的.主要包括: 标准流类:处理与标准输入设备(键盘)和输出设备(显示器)关联的数据流 文件流类:处理与磁盘文件关联的数据流 字符串流类:利用内存中的字符数组处理数据的输入输出 异常类等:处理异常错误. 标准IO对象: 包含iostream类的C++程序启动时,将

JavaSE应用程序打包成可运行程序

JavaSE应用程序打包成可运行程序 所需软件 - MyEclipse或Eclipse(如会打包命令,此软件可省略) - exe4j 实现过程: 1.使用MyEclipse进行程序打包. - 选中项目右键 Export... 选中 JAR File...点击next 选择输出文件位置. 选择下一步取消勾选Export classs files with compile errors 选择下一步,选择包含main方法的类,之后点击finish就生成jar包了 2.生成windows下可运行的程序(

将EXE安装包封装成MSI应用程序

将EXE安装包封装成MSI应用程序 我们在使用GPO进行软件分发时,可能会遇到这样的情况:我们希望软件在安装过程中无需与用户进行交互,但软件的原有的安装程序不支持,比如.exe安装包.我们考虑使用第三方工具为这些不符合条件的安装程序重新进行打包,将其做成MSI类型的安装文件,再利用组策略进行分发. 我在这里使用scalable公司的SMART PACKAGER PRO教大家如何制作MSI应用程序,此应用程序可以到www.scalable.com网站去下载,使用SMART PACKAGER PRO