在阅读别人开发的项目中,也许你会经常看到了多处使用异常的代码,也许你也很少遇见使用异常处理的代码。那在什么时候该使用异常,又在什么时候不该使用异常呢?在学习完异常基本概念和语法之后,后面会有讲解。
(1)异常抛出和捕捉语句 //1.抛出异常 throw 异常对象 //2.异常捕捉 try{ 可能会发生异常的代码 }catch(异常对象){ 异常处理代码 }throw子句:throw 子句用于抛出异常,被抛出的异常可以是C++的内置类型(例如: throw int(1);),也可以是自定义类型。
try区段:这个区段中包含了可能发生异常的代码,在发生了异常之后,需要通过throw抛出。
catch子句:每个catch子句都代表着一种异常的处理。catch子句用于处理特定类型的异常。catch块的参数推荐采用地址传递而不是值传递,不仅可以提高效率,还可以利用对象的多态性。
(2)异常的处理规则throw抛出的异常类型与catch抓取的异常类型要一致;
throw抛出的异常类型可以是子类对象,catch可以是父类对象;
catch块的参数推荐采用地址传递而不是值传递,不仅可以提高效率,还可以利用对象的多态性。另外,派生类的异常捕获要放到父类异常扑获的前面,否则,派生类的异常无法被扑获;
如果使用catch参数中,使用基类捕获派生类对象,一定要使用传递引用的方式,例如catch (exception &e);
异常是通过抛出对象而引发的,该对象的类型决定了应该激活哪个处理代码;
被选中的处理代码是调用链中与该对象类型匹配且离抛出异常位置最近的那一个;
在try的语句块内声明的变量在外部是不可以访问的,即使是在catch子句内也不可以访问;
栈展开会沿着嵌套函数的调用链不断查找,直到找到了已抛出的异常匹配的catch子句。如果抛出的异常一直没有函数捕获(catch),则会一直上传到c++运行系统那里,导致整个程序的终止。
(3)实例实例1:抛出自定义类型异常。
class Data { public: Data() {} }; void fun(int n) { if(n==0) throw 0;//抛异常 int异常 if(n==1) throw "error"; //抛字符串异常 if(n==2) { Data data; throw data; } if(n>3) { throw 1.0; } } int main() { try { fun(6);//当异常发生fun里面,fun以下代码就不会再执行,调到catch处执行异常处理代码,后继续执行catch以外的代码。当throw抛出异常后,没有catch捕捉,则整个程序会退出,不会执行整个程序的以下代码 cout<<"*************"<<endl; }catch (int i) { cout<<i<<endl; }catch (const char *ptr) { cout<<ptr<<endl; }catch(Data &d) { cout<<"data"<<endl; }catch(...)//抓取 前面异常以外的所有其他异常 { cout<<"all"<<endl; } return 0; }实例2:标准出错类抛出和捕捉异常。
#include <iostream> using namespace std; int main() { try { char* p = new char[0x7fffffff]; //抛出异常 } catch (exception &e){ cout << e.what() << endl; //捕获异常,然后程序结束 } return 0; }输出结果:
当使用new进行开空间时,申请内存失败,系统就会抛出异常,不用用户自定义异常类型,此时捕获到异常时,就可告诉使用者是哪里的错误,便于修改。
实例3:继承标准出错类的派生类的异常抛出和捕捉。
#include <iostream> #include <exception> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> using namespace std; class FileException :public exception { public: FileException(string msg) { this->exStr = msg; } virtual const char*what() const noexcept//声明这个函数不能再抛异常 { return this->exStr.c_str(); } protected: string exStr; }; void fun() { int fd = ::open("./open.txt",O_RDWR); if(fd<0) { FileException openFail("open fail"); //创建异常对象 throw openFail;//抛异常 } } int main( ) { try { fun(); } catch (exception &e) {//一般需要使用引用 cout<<e.what()<<endl; } cout<<"end"<<endl; return 0; }当文件不存在时,输出结果: