自定义C++异常处理

例1:自定义一个继承自excepton的异常类myException

C++标准中,定义在<stdexcept>中的任何异常类都派生自exception Class,本例也只是简单地由exception继承,在try段抛出一个异常并捕捉。代码如下:

  1. /*++ test.cpp 
  2. version:1.0 
  3. decript:define a exception class named myException 
  4.         derived from base class exception 
  5.         which is declared in <exception> 
  6. created:2011-08-14 
  7. author: btwsmile 
  8. --*/  
  9. #include<exception>  
  10. #include<iostream>  
  11. using namespace std;  
  12.   
  13. //customized exception class 'myException'  
  14. class myException:public exception  
  15. {  
  16. public:  
  17.     myException():exception("ERROR! Don't divide a number by integer zero.\n")  
  18.     {  
  19.     }  
  20. };  
  21. //entry of the application  
  22. int main()  
  23. {  
  24.     int x=100,y=0;  
  25.     try  
  26.     {  
  27.         if(y==0) throw myException();  
  28.         else cout<<x/y;  
  29.     }  
  30.     catch(myException& me)  
  31.     {  
  32.         cout<<me.what();  
  33.     }  
  34.     system("pause");  
  35.     return 0;  
  36. }  

结果如下:
ERROR! Don't divide a number by integer zero.
请按任意键继续. . .                                                  

显然,异常被捕捉到了。此处需要说明的是,VC对异常处理类exception进行了扩展,本例之所以能够使用exception("ERROR!....")的初始化方法正出于这样的原因,C++标准是不允许这样做的。

与此同时,VC又没有遵循标准,有力地支持terminate和unexpected,它只保留了语法,却在编译运行时不提供支持。为了结合terminate和unexpected更加深入了解C++的异常处理,下面的例子采用Dev cpp IDE实现。

例2:依照C++标准实现自定义异常类myException并将throw语句封装到函数check()中

涉及到的更改正如标题所述,(1)重写基类的what()函数,返回错误信息;(2)将throw myException()封装到check()函数中;(3)允许check()函数抛出myException类型的异常。代码如下:

  1. /*++ test.cpp 
  2. version:1.1 
  3. decript:define a exception class named myException  
  4.         according to C++ standard, 
  5.         derived from base class exception 
  6.         which is declared in <exception> 
  7.         !also,encapusulate throw into a function 
  8. created:2011-08-14 
  9. author: btwsmile 
  10. --*/  
  11. #include<exception>  
  12. #include<iostream>  
  13. using namespace std;  
  14.   
  15. //customized exception class 'myException'  
  16. class myException:public exception  
  17. {  
  18. public:  
  19.    const char* what()const throw()//#1   
  20.    {  
  21.         return "ERROR! Don't divide a number by integer zero.\n";  
  22.    }      
  23. };  
  24. void check(int y) throw(myException)//#2  
  25. {  
  26.      if(y==0) throw myException();  
  27. }  
  28. //entry of the application  
  29. int main()  
  30. {  
  31.     int x=100,y=0;  
  32.     try  
  33.     {  
  34.         check(y);  
  35.         cout<<x/y;  
  36.     }  
  37.     catch(myException& me)  
  38.     {  
  39.         cout<<me.what();  
  40.     }  
  41.     system("pause");  
  42.     return 0;  
  43. }  

结果与例1完全相同。需说明的是,紧跟check()后的throw列表表明允许该函数抛出的异常类型。这里不得不产生疑问,如果抛出了一个不被允许的异常类型将怎样?

例3:抛出unexpected异常

check函数体之后的throw列表,规定了允许抛出的异常类型,一旦违背,就将触发unexpected。可以把unexpected看作系统自动调用的CALLBACK函数,不同的是,也可以手工触发它的执行。本例的情况属于前者。代码如下:

 

  1. /*++ test.cpp 
  2. version:1.3 
  3. decript:define an unexpected excption handler, 
  4.         set it by using set_unexpected, 
  5.         modify the throw list of function check 
  6. created:2011-08-14 
  7. author: btwsmile 
  8. --*/  
  9. #include<exception>  
  10. #include<iostream>  
  11. using namespace std;  
  12.   
  13. //customized exception class 'myException'  
  14. class myException:public exception  
  15. {  
  16. public:  
  17.    const char* what()const throw()  
  18.    {  
  19.         return "ERROR! Don't divide a number by integer zero.\n";  
  20.    }      
  21. };  
  22. void check(int y) throw()//#1 only int-type exception is permitted  
  23. {  
  24.      if(y==0) throw myException();  
  25. }  
  26. void myUnexpected()  
  27. {  
  28.      cout<<"Unexpected exception caught!\n";  
  29.      system("pause");  
  30.      exit(-1);  
  31. }  
  32. //entry of the application  
  33. int main()  
  34. {  
  35.     unexpected_handler oldHandler=set_unexpected(myUnexpected);  
  36.     int x=100,y=0;  
  37.     try  
  38.     {  
  39.         check(y);  
  40.         cout<<x/y;  
  41.     }  
  42.     catch(myException& me)  
  43.     {  
  44.         cout<<me.what();  
  45.     }  
  46.     system("pause");  
  47.     return 0;  
  48. }  

结果如下:

Unexpected exception caught!
请按任意键继续. . .                    

check函数的throw列表为空,即不允许抛出任何类型的异常,然而实际上当异常发生时,系统不能等闲视之,它将调用unexpected处理方法。所以,限定一个函数throw列表为空是值得程序员警醒的事,需要特别留意。如果将#1处的代码修改为throw(int)等也能得到相同的结果。所谓unexpected异常,说白了就是函数体允许抛出异常类型范围之外的异常。如果check函数后面根本没有throw,则表示函数任何类型的异常都被允许。

例4:抛出函数体允许的异常,但没被捕捉到的情况

思考这样一个问题,如果函数check的throw列表中有异常类型myException,而且在y==0时,它的确抛出myException类型的异常,但是没有被catch到,这时会发生什么?

在正式回答这个问题之前,先讨论“没被catch到”的意思。比如,修改例3的代码如下:(##为修改之处)

  1. /*++ test.cpp 
  2. version:1.4.1 
  3. decript: 
  4.         how to understand "exception not caucht"? 
  5. created:2011-08-14 
  6. author: btwsmile 
  7. --*/  
  8. #include<exception>  
  9. #include<iostream>  
  10. using namespace std;  
  11.   
  12. //customized exception class 'myException'  
  13. class myException:public exception  
  14. {  
  15. public:  
  16.    const char* what()const throw()  
  17.    {  
  18.         return "ERROR! Don't divide a number by integer zero.\n";  
  19.    }      
  20. };  
  21. void check(int y) //any type of exception is permitted  
  22. {  
  23.      if(y==0) throw myException();  
  24. }  
  25. void myUnexpected()  
  26. {  
  27.      cout<<"Unexpected exception caught!\n";  
  28.      system("pause");  
  29.      exit(-1);  
  30. }  
  31. //entry of the application  
  32. int main()  
  33. {  
  34.     unexpected_handler oldHandler=set_unexpected(myUnexpected);  
  35.     int x=100,y=0;  
  36.     try  
  37.     {  
  38.         check(y);  
  39.         cout<<x/y;  
  40.     }  
  41.     catch(int &e) //##1 no catch sentence matches the throw type  
  42.     {  
  43.         cout<<e<<endl;  
  44.     }  
  45.     /*               ##2 if add this part, any type which's not handler before will 
  46.                         be caught 
  47.     catch(...) 
  48.     { 
  49.                     cout<<"Unkown exception caught!\n"; 
  50.          } 
  51.     */  
  52.     system("pause");  
  53.     return 0;  
  54. }  

编译运行,程序将会出错,因为check函数抛出的myException异常没有被处理。在缺省情况下,一旦出现抛出异常没被处理的问题,系统将自动调用abort()函数,终止程序允许,在控制台将会看到这样的提示:
This application has requested the Runtime to terminate it in an unusual way.Please contact the application's support team for more information.

不过可以增加##2部分的代码,catch(...)表示捕捉任何类型的异常。

注意:check函数不被允许的异常类型并不会进入到catch语句的判断中来,因此catch(...)对unexpected exception没有作用。

仍然考虑没有##2部分的情况。正如前面所述,系统将自动调用abort()函数终止程序。实际上,它触发的是terminate,类似于unexpected,仍然可以自定义terminate的处理方法。甚至terminate语法上跟unexpected都十分近似。修改代码为:

  1. /*++ test.cpp 
  2. version:1.4.2 
  3. decript: 
  4.         how to understand "exception not caucht"? 
  5. created:2011-08-14 
  6. author: btwsmile 
  7. --*/  
  8. #include<exception>  
  9. #include<iostream>  
  10. using namespace std;  
  11.   
  12. //customized exception class 'myException'  
  13. class myException:public exception  
  14. {  
  15. public:  
  16.    const char* what()const throw()  
  17.    {  
  18.         return "ERROR! Don't divide a number by integer zero.\n";  
  19.    }      
  20. };  
  21. void check(int y) //any type of exception is permitted  
  22. {  
  23.      if(y==0) throw myException();  
  24. }  
  25. void myUnexpected()  
  26. {  
  27.      cout<<"Unexpected exception caught!\n";  
  28.      system("pause");  
  29.      exit(-1);  
  30. }  
  31. void myTerminate() //##1 set it be the terminate handler  
  32. {  
  33.      cout<<"Unhandler exception!\n";  
  34.      system("pause");  
  35.      exit(-1);  
  36. }  
  37. //entry of the application  
  38. int main()  
  39. {  
  40.     unexpected_handler oldHandler=set_unexpected(myUnexpected);  
  41.     terminate_handler preHandler=set_terminate(myTerminate);  
  42.     int x=100,y=0;  
  43.     try  
  44.     {  
  45.         check(y);  
  46.         cout<<x/y;  
  47.     }  
  48.     catch(int &e) //no catch sentence matches the throw type  
  49.     {  
  50.         cout<<e<<endl;  
  51.     }  
  52.     system("pause");  
  53.     return 0;  
  54. }  

结果如下:

Unhandler exception!
请按任意键继续. . .    

结论:C++为异常处理提供了友好的支持。

用户可以自定义异常类型,异常类型并不受到限制,可以是内建数据类型如int,double等,也可以是自定义的类,也可以从C++某个异常类继承下来。例1采用了派生自exception的方法。

除此之外,在定义函数时,可以显式指定函数体抛出的异常类型。隐式情况下,缺省允许函数抛出任何类型的异常。有可以增加throw语句,对异常类型加以限制。特别的是,throw()表示不允许函数抛出任何类型的异常。如果违反了throw列表规定的异常类型,系统将调用unexpected hanlder进行处理,可以自定义unexpected异常处理方法。例2和例3对它们进行了说明。

如果对于函数体throw列表合法的异常被抛出,但是却没有被程序捕捉处理,系统将调用terminate handler进行处理。缺省情况下,只是简单调用abort()函数终止程序,同样可以自定义terminate处理方法。例4对它进行了说明。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值