学习用于异常处理的terminate()函数
来源:互联网 发布:网易中超数据库 编辑:程序博客网 时间:2024/05/21 08:39
异常处理是一个微妙的问题,你应该尽可能优雅地处理所有异常。要达到这个目的,你需要学习terminate()函数。
#include 〈exception〉 void on_terminate() int main() 避免这种情形的方案一开始看起来很简单: int main() 然而terminate()函数在许多其它情况下会被调用,包括: 当你抛出一个异常,并且在它的拷贝构造函数中,另一个异常被抛出。 #include 〈iostream〉 void on_terminate() //////////////////////////////// [1] void case_1() //////////////////////////////// [2] void case_2() //////////////////////////////// [3] void case_3() //////////////////////////////// [4] void case_4() //////////////////////////////// [5] //////////////////////////////// [6] void func_violating_exception_specification_6() throw(std::exception) // 注意:按照我们的例子,在这个函数中我们只应该抛出 void case_6() //////////////////////////////// [7] void func_violating_exception_specification_7() throw(std::exception) void case_7() int main() std::ostream& get_log() { /* code */ } void on_terminate() int main()
terminate()函数在程序抛出一个异常并且异常没有被捕获的时候被调用,像下面这样:
#include 〈iostream〉
{
std::cout 〈〈 "terminate() 函数被调用了!" 〈〈 std::endl;
std::cin.get();
}
{
// 如果用 VC6,去掉“std::”前缀
std::set_terminate( on_terminate);
throw std::exception();
std::cout 〈〈 "terminate() 函数没有被调用!" 〈〈 std::endl;
std::cin.get();
return 0;
}
{
try
{
/* code */
}
catch( std::exception & exc)
{
// 记录到日志,或作其他处理
}
catch(...)
{
// 记录下“Unknown exception”
}
return 0;
}
不过,在多线程应用程序中情况变得有点复杂,因为你创建的每个线程都要有上面的(catch)处理过程。
在堆栈展开的过程中抛出一个异常,此时析构函数抛出一个异常。
当一个静态对象的构造函数或析构函数抛出异常时。
当一个用atexit注册过的函数抛出一个异常时。
当你在代码中写下“throw;”(这意味着重新抛出当前异常),然而并没有当前异常时。
当一个函数抛出一个它的异常说明不允许的异常时
当默认的unexpected()处理过程被调用时
下面的代码演示了上面各种情况下的结果:
#include 〈exception〉
{ std::cout 〈〈 "terminate()函数被调用了!" 〈〈 std::endl;
std::cin.get(); }
struct custom_exception
{
custom_exception() {}
custom_exception( const custom_exception &)
{ throw std::exception(); }
};
{
try
{ throw custom_exception(); }
catch(...)
{}
}
struct throw_in_destructor
{
~throw_in_destructor() { throw std::exception(); }
};
{
try
{
throw_in_destructor temp;
throw std::exception();
}
catch(...)
{}
}
struct static_that_throws
{
static_that_throws() { throw std::exception(); }
};
{
// 注意:用try/catch块包围下面的代码并不起作用
static static_that_throws obj;
}
void throw_at_exit()
{ throw std::exception(); }
{ atexit( throw_at_exit); }
void case_5()
{ throw; }
class custom_6_a {};
class custom_6_b {};
{ throw custom_6_a(); }
// std::exception(在函数func_violating_exception_specification
// 的定义中说明的异常);但我们没有这样做,
// 因此,terminate() 被调用
void on_unexpected()
{ throw custom_6_b(); }
{
std::set_unexpected( on_unexpected);
try
{ func_violating_exception_specification_6(); }
catch(...)
{}
}
class custom_7 {};
{ throw custom_7(); }
{
try
{ func_violating_exception_specification_7(); }
catch(...)
{}
}
{
std::set_terminate( on_terminate);
// 注意:确保每次仅去掉下面一个调用的注释,
// 以便分析时将每种情况隔离开来
case_1();
// case_2();
// case_3();
// case_4();
// case_5();
// case_6();
// case_7();
return 0;
}
尽管你应该努力避免terminate()函数会被调用的情况,我们还是建议你创建自己的terminate()处理过程。你的处理过程要做的唯一合理的事是记录一条消息到日志中。不管怎样,确保你的日志不会抛出任何异常。
{
std::ostream & log = get_log();
// 确保我们不会抛出任何东西!
try
{
log.exceptions( std::ios_base::goodbit);
}
catch (...)
{}
log 〈〈 "terminate() 被调用了!" 〈〈 std::endl;
}
{
std::set_terminate( on_terminate) ;
// . . .
}
- 学习用于异常处理的terminate()函数
- 学习用于异常处理的terminate()函数
- C++ 异常处理 terminate函数使用
- C++异常处理之terminate函数
- C++异常处理之terminate函数
- std的terminate函数
- 一个用于处理分隔","的函数tf_split_str
- 8.2 用于处理函数的习惯用语
- 8.2 用于处理函数的习惯用语
- 一个用于J2EE应用程序的异常处理框架
- 一个用于J2EE应用程序的异常处理框架
- 一个用于J2EE应用程序的异常处理框架
- 一个用于J2EE应用程序的异常处理框架
- 个用于J2EE应用程序的异常处理框架
- 第十七章 用于大型程序的工具(异常处理)
- 一个用于J2EE应用程序的异常处理框架
- C++ primer第二次阅读学习笔记(第17章:用于大型程序的工具:异常处理)
- 异常一般用于突发情况的处理,而不是用于正常的业务逻辑判断
- jfreechart在linux环境下显示中文为乱码的解决办法
- Overloading unary operators (C++ only)
- ASP.NET MVC 重点教程一周年版 第六回 过滤器Filter
- RTTI特性小究(dynamic_cast转换操作符和typeid操作符)
- ASP.NET MVC 重点教程一周年版 第七回 UrlHelper
- 学习用于异常处理的terminate()函数
- 有道阅读,想自己看别人
- 使用word2007发布csdn博客
- Flyer test
- linux学习笔记
- 统一论:3G手机、云计算、SaaS、业务开发平台、SOA、BPEL
- C#正则表达式参考
- moodle/网页 中 添加音视频的代码
- Moodle安装完全手册