您的位置:首页 > 编程语言 > C语言/C++

c++异常处理机制示例及讲解

2010-08-18 09:18 507 查看
下面的代码直接贴到你的console工程中,可以运行调试看看效果,并分析c++的异常机制。
#include "stdafx.h"
#include
#include
#include
// 内存泄露检测机制
#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;
}
知识点: c++异常机制
一、 概述
C++自身有着非常强的纠错能力,发展到如今,已经建立了比较完善的异常处理机制。C++的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句,函数,结构和类,致使编译程序无法进行。另一种是运行时发生的错误,一般与算法有关。
关于语法错误,不必多说,写代码时心细一点就可以解决。C++编译器的报错机制可以让我们轻松地解决这些错误。
第二种是运行时的错误,常见的有文件打开失败、数组下标溢出、系统内存不足等等。而一旦出现这些问题,引发算法失效、程序运行时无故停止等故障也是常有的。这就要求我们在设计软件算法时要全面。比如针对文件打开失败的情况,保护的方法有很多种,最简单的就是使用“return”命令,告诉上层调用者函数执行失败;另外一种处理策略就是利用c++的异常机制,抛出异常。
二、c++异常处理机制
    C++异常处理机制是一个用来有效地处理运行错误的非常强大且灵活的工具,它提供了更多的弹性、安全性和稳固性,克服了传统方法所带来的问题.
    异常的抛出和处理主要使用了以下三个关键字: try、 throw 、 catch 。
    抛出异常即检测是否产生异常,在C++中,其采用throw语句来实现,如果检测到产生异常,则抛出异常。该语句的格式为:
    throw 表达式;
    如果在try语句块的程序段中(包括在其中调用的函数)发现了异常,且抛弃了该异常,则这个异常就可以被try语句块后的某个catch语句所捕获并处理,捕获和处理的条件是被抛弃的异常的类型与catch语句的异常类型相匹配。由于C++使用数据类型来区分不同的异常,因此在判断异常时,throw语句中的表达式的值就没有实际意义,而表达式的类型就特别重要。
try-catch语句形式如下 :



try


{


        包含可能抛出异常的语句;


}


catch(类型名 [形参名]) // 捕获特定类型的异常


{



}


catch(类型名 [形参名]) // 捕获特定类型的异常


{



}


catch(...)    // 三个点则表示捕获所有类型的异常


{


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


1        #include     //包含头文件


2        #include



   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 : "<
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: