#include <iostream> #include <crtdbg.h> using namespace std; //内存泄露机制检测 #define _CRTDBG_MAP_ALLOC #ifndef _DEBUG #define new new (_NORMAL_BLOCK,_FILE_,_LINE_) #endif class MyException{ public: MyException(int errorID) { cout<<"MyException is called"<<endl; errorNum=errorID; } MyException ( MyException &my) { cout<<"copy constructor is called "<<endl; this->errorNum=my.getErrorID(); } ~MyException() { cout<<"~MyException is called "<<endl; } int getErrorID() { return errorNum; } private: int errorNum; }; int main() { _CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF |_CRTDBG_LEAK_CHECK_DF); int throwErrorID=110; cout<<"input test code :"<<endl; cin>>throwErrorID; try { if (throwErrorID==110) { MyException myEx(110); /* 抛出对象地址,由catch (MyException * e) 捕获,对象传递是地址不会调用复制构造函数。 传地址是提倡方法,不会频繁调用对象构造和复制构造函数 catch 执行后,myEx析构 */ throw &myEx; }else if (throwErrorID==119) { MyException myEx(119); /* 抛出对象,通过复制构造函数创建临时对象传给catch 由catch (MyException e)捕获, catch语句再次调用复制构造函数创建临时对象复制这里传过去 的对象,throw后 myEx析构 */ throw myEx; } else if (throwErrorID==120) { /* 不提倡这样做法, catch (MyException * e) 不执行delete 会发生内存的泄露。 */ MyException *myEx=new MyException(120); throw myEx; } else { /* 直接创建对象抛出 相当于临时对象传递给catch语句 catch接收时通过复制构造函数再次创建临时变量接受传递过去对象 throw后两次创建对象被析构掉 */ throw MyException(throwErrorID); } } catch (MyException * e) { cout<<"执行了MyException *myEx"<<endl; cout<<"errorID :"<<e->getErrorID()<<endl; //delete e; } catch (MyException e) { cout<<"执行了MyException myEx"<<endl; cout<<"errorID :"<<e.getErrorID()<<endl; } catch (...) { cout<<"执行了catch(...)"<<endl; throw; } getchar(); }
知识点: c++异常机制
一、 概述
C++自身有着非常强的纠错能力,发展到如今,已经建立了比较完善的异常处理机制。C++的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句,函数,结构和类,致使编译程序无法进行。另一种是运行时发生的错误,一般与算法有关。
关于语法错误,不必多说,写代码时心细一点就可以解决。C++编译器的报错机制可以让我们轻松地解决这些错误。
第二种是运行时的错误,常见的有文件打开失败、数组下标溢出、系统内存不足等等。而一旦出现这些问题,引发算法失效、程序运行时无故停止等故障也是常有的。这就要求我们在设计软件算法时要全面。比如针对文件打开失败的情况,保护的方法有很多种,最简单的就是使用“return”命令,告诉上层调用者函数执行失败;另外一种处理策略就是利用c++的异常机制,抛出异常。
二、c++异常处理机制
C++异常处理机制是一个用来有效地处理运行错误的非常强大且灵活的工具,它提供了更多的弹性、安全性和稳固性,克服了传统方法所带来的问题.
异常的抛出和处理主要使用了以下三个关键字: try、 throw 、 catch 。
抛出异常即检测是否产生异常,在C++中,其采用throw语句来实现,如果检测到产生异常,则抛出异常。该语句的格式为:
throw 表达式;
如果在try语句块的程序段中(包括在其中调用的函数)发现了异常,且抛弃了该异常,则这个异常就可以被try语句块后的某个catch语句所捕获并处理,捕获和处理的条件是被抛弃的异常的类型与catch语句的异常类型相匹配。由于C++使用数据类型来区分不同的异常,因此在判断异常时,throw语句中的表达式的值就没有实际意义,而表达式的类型就特别重要。
【范例1】处理除数为0的异常。该范例将上述除数为0的异常可以用try/catch语句来捕获异常,并使用throw语句来抛出异常,从而实现异常处理,实现代码如代码
这是利用系统的Exception 类处理。
#include <iostream> #include <stdlib.h> using namespace std; double fuc(double x,double y) { if (y==0) { throw y; } return x/y; } int main() { double res; try{ res=fuc(4,2); cout<<"result is "<<res<<endl; res=fuc(2,0); } catch(double) { cerr<<"error\n"; exit(1); } return 0; }
【范例2】自定义异常类型 (在本文开始的代码中已经给出示范)
三、异常的接口声明
为了加强程序的可读性,使函数的用户能够方便地知道所使用的函数会抛出哪些异常,可以在函数的声明中列出这个函数可能抛出的所有异常类型,例如:
这表明函数fun()可能并且只可能抛出类型(A,B,C,D)及其子类型的异常。
如果在函数的声明中没有包括异常的接口声明,则此函数可以抛出任何类型的异常,例如:
一个不会抛出任何类型异常的函数可以进行如下形式的声明:
五、异常处理中需要注意的问题
1. 如果抛出的异常一直没有函数捕获(catch),则会一直上传到c++运行系统那里,导致整个程序的终止
2. 一般在异常抛出后资源可以正常被释放,但注意如果在类的构造函数中抛出异常,系统是不会调用它的析构函数的,处理方法是:如果在构造函数中要抛出异常,则在抛出前要记得删除申请的资源。
3. 异常处理仅仅通过类型而不是通过值来匹配的,所以catch块的参数可以没有参数名称,只需要参数类型。
4. 函数原型中的异常说明要与实现中的异常说明一致,否则容易引起异常冲突。
5. 应该在throw语句后写上异常对象时,throw先通过Copy构造函数构造一个新对象,再把该新对象传递给 catch.
那么当异常抛出后新对象如何释放?
异常处理机制保证:异常抛出的新对象并非创建在函数栈上,而是创建在专用的异常栈上,因此它才可以跨接多个函数而传递到上层,否则在栈清空的过程中就会被销毁。所有从try到throw语句之间构造起来的对象的析构函数将被自动调用。但如果一直上溯到main函数后还没有找到匹配的catch块,那么系统调用terminate()终止整个程序,这种情况下不能保证所有局部对象会被正确地销毁。
6. catch块的参数推荐采用地址传递而不是值传递,不仅可以提高效率,还可以利用对象的多态性。另外,派生类的异常扑获要放到父类异常扑获的前面,否则,派生类的异常无法被扑获。
7. 编写异常说明时,要确保派生类成员函数的异常说明和基类成员函数的异常说明一致,即派生类改写的虚函数的异常说明至少要和对应的基类虚函数的异常说明相同,甚至更加严格,更特殊。