`

C++ _try _catch()的妙处(收藏)

    博客分类:
  • c++
 
阅读更多
转: http://hi.baidu.com/wind_stay/blog/item/1c4f311f54af761540341729.html
以前都是用try{} catch(…){}来捕获C++中一些意想不到的异常, 今天看了Winhack的帖子才知道,这种方法在VC中其实是靠不住的。例如下面的代码:

try
{
BYTE* pch ;
pch = ( BYTE* )00001234 ;   //给予一个非法地址
*pch = 6 ; //对非法地址赋值,会造成Access Violation 异常
}
catch(...)
{
AfxMessageBox( "catched" ) ;
}
这段代码在debug下没有问题,异常会被捕获,会弹出”catched”的消息框。 但在Release方式下如果选择了编译器代码优化选项,则VC编译器会去搜索try块中的代码, 如果没有找到throw代码, 他就会认为try catch结构是多余的, 给优化掉。 这样造成在Release模式下,上述代码中的异常不能被捕获,从而迫使程序弹出错误提示框退出。

那么能否在release代码优化状态下捕获这个异常呢, 答案是有的。 就是__try, __except结构, 上述代码如果改成如下代码异常即可捕获。

__try
{
BYTE* pch ;
pch = ( BYTE* )00001234 ;   //给予一个非法地址
*pch = 6 ; //对非法地址赋值,会造成Access Violation 异常
}
__except( EXCEPTION_EXECUTE_HANDLER )
{
AfxMessageBox( "catched" ) ;
}
但是用__try, __except块还有问题, 就是这个不是C++标准, 而是Windows平台特有的扩展。 而且如果在使用过程中涉及局部对象析构函数的调用,则会出现C2712 的编译错误。 那么还有没有别的办法呢?

当然有, 就是仍然使用C++标准的try{}catch(..){}, 但在编译命令行中加入 /EHa 的参数。这样VC编译器不会把try catch模块给优化掉了。

找到一篇比较好的英文文章谈这个问题: http://members.cox.net/doug_web/eh.htm

用C++10 年多了 , 居然这么基础的问题都搞错, 真是汗颜。 要加紧学习啊, Stay Hungry, Stay Foolish!

Written by oldmonk on 九月 11th, 2006 with 2 comments.
Read more articles on IT.



http://se.csai.cn/ExpertEyes/No138.htm
                                     C++中catch(…)如何使用
上一篇文章中详细讲了讲C++异常处理模型的trycatch使用语法,其中catch关键字是用来定义catch block的,它后面带一个参数,用来与异常对象的数据类型进行匹配。注意catch关键字只能定义一个参数,因此每个catch block只能是一种数据类型的异常对象的错误处理模块。如果要想使一个catch block能抓获多种数据类型的异常对象的话,怎么办?C++标准中定义了一种特殊的catch用法,那就是” catch(…)”。
感性认识

1、catch(…)到底是一个什么样的东东,先来个感性认识吧!看例子先:

int main()
{
try
{
cout << "在 try block 中, 准备抛出一个异常." << endl;
//这里抛出一个异常(其中异常对象的数据类型是int,值为1)
throw 1;
}
//catch( int& value )
//注意这里catch语句
catch( …)
{
cout << "在 catch(…) block 中, 抛出的int类型的异常对象被处理" << endl;
}
}

  2、哈哈!int类型的异常被catch(…)抓获了,再来另一个例子:

int main()
{
try
{
cout << "在 try block 中, 准备抛出一个异常." << endl;
//这里抛出一个异常(其中异常对象的数据类型是double,值为0.5)
throw 0.5;
}
//catch( double& value )
//注意这里catch语句
catch( …)
{
cout << "在 catch(…) block 中, double类型的异常对象也被处理" << endl;
}
}

   3、同样,double类型的异常对象也被catch(…)块抓获了。是的,catch(..)能匹配成功所有的数据类型的异常对象,包括C++语言提 供所有的原生数据类型的异常对象,如int、double,还有char*、int*这样的指针类型,另外还有数组类型的异常对象。同时也包括所有自定义 的抽象数据类型。例程如下:

int main()
{
try
{
cout << "在 try block 中, 准备抛出一个异常." << endl;
//这里抛出一个异常(其中异常对象的数据类型是char*)
char* p=0;
throw p;
}
//catch( char* value )
//注意这里catch语句
catch( …)
{
cout << "在 catch(…) block 中, char*类型的异常对象也被处理" << endl;
}
}


int main()
{
try
{
cout << "在 try block 中, 准备抛出一个异常." << endl;
//这里抛出一个异常(其中异常对象的数据类型是int[])
int a[4];
throw a;
}
//catch( int value[] )
//注意这里catch语句
catch( …)
{
cout << "在 catch(…) block 中, int[]类型的异常对象也被处理" << endl;
}
}

  4、对于抽象数据类型的异常对象。catch(…)同样有效,例程如下:

class MyException
{
public:
protected:
int code;
};

int main()
{
try
{
cout << "在 try block 中, 准备抛出一个异常." << endl;
//这里抛出一个异常(其中异常对象的数据类型是MyException)
throw MyException();
}
//catch(MyException& value )
//注意这里catch语句
catch( …)
{
cout << "在catch(…) block中, MyException类型的异常对象被处理" << endl;
}
}
对catch(…)有点迷糊?
1、究竟对catch(…)有什么迷糊呢?还是看例子先吧!
void main()
{
int* p = 0;

try
{
// 注意:下面这条语句虽然不是throw语句,但它在执行时会导致系统
// 出现一个存储保护错误的异常(access violation exception)
*p = 13; // causes an access violation exception;
}
catch(...)
{
//catch(…)能抓获住上面的access violation exception异常吗?
cout << "在catch(…) block中" << endl;
}
}

  请问上面的程序运行时会出现什么结果吗?catch(…)能抓获住系统中出现的access violation exception异常吗?朋友们!和我们的主人公阿愚一样,自己动手去测试一把!
结果又如何呢?实际上它有两种不同的运行结果,在window2000系统下用VC来测试运行这个小程序时,发现程序能输出"在catch(…) block中"的语句在屏幕上,也即catch(…) 能成功抓获住系统中出现的access violation exception异常,很厉害吧!但如果这个同样的程序在linux下用gcc编译后运行时,程序将会出现崩溃,并在屏幕上输出”segment fault”的错误信息。

主人公阿愚有点急了,也开始有点迷糊了,为什么?为什么?为什么同样一个程序在两种不同的系统上有不同的表现呢?其原因就是:对于这种由于硬件或操作 系统出现的系统异常(例如说被零除、内存存储控制异常、页错误等等)时,window2000系统有一个叫做结构化异常处理(Structured Exception Handling,SEH)的机制,这个东东太厉害了,它能和VC中的C++异常处理模型很好的结合上(实际上VC实现的C++异常处理模型很大程度上建 立在SEH机制之上的,或者说它是SEH的扩展,后面文章中会详细阐述并分析这个久富盛名的SEH,看看catch(…)是如何神奇接管住这种系统异常出 现后的程序控制流的,不过这都是后话)。而在linux系统下,系统异常是由信号处理编程方法来控制的(信号处理编程,signal processing progamming。在介绍unix和linux下如何编程的书籍中,都会有对信号处理编程详细的介绍,当然执著的主人公阿愚肯定对它也不会放过,会深 入到unix沿袭下来的信号处理编程内部的实现机制,并尝试完善改进它,使它也能够较好地和C++异常处理模型结合上)。

那么C++标准中对于这种同一个程序有不同的运行结果有何解释呢?这里需要注意的是,window2000系统下catch(…)能捕获住系统异常, 这完全是它自己的扩展。在C++标准中并没有要求到这一点,它只规定catch(…)必须能捕获程序中所有通过throw语句抛出的异常。因此上面的这个 程序在linux系统下的运行结果也完全是符合C++标准的。虽然大家也必须承认window2000系统下对C++异常处理模型的这种扩展确实是一个很 不错的完善,极大得提高了程序的安全性。

为什么要用catch(…)这个东东?

程序员朋友们也许会说,这还有问吗?这篇文章的一开始不就讲到了吗?catch(…)能够捕获多种数据类型的异常对象,所以它提供给程序员一种对异常 对象更好的控制手段,使开发的软件系统有很好的可靠性。因此一个比较有经验的程序员通常会这样组织编写它的代码模块,如下:

void Func()
{
try
{
// 这里的程序代码完成真正复杂的计算工作,这些代码在执行过程中
// 有可能抛出DataType1、DataType2和DataType3类型的异常对象。
}
catch(DataType1& d1)
{
}
catch(DataType2& d2)
{
}
catch(DataType3& d3)
{
}
// 注意上面try block中可能抛出的DataType1、DataType2和DataType3三
// 种类型的异常对象在前面都已经有对应的catch block来处理。但为什么
// 还要在最后再定义一个catch(…) block呢?这就是为了有更好的安全性和
// 可靠性,避免上面的try block抛出了其它未考虑到的异常对象时导致的程
// 序出现意外崩溃的严重后果,而且这在用VC开发的系统上更特别有效,因
// 为catch(…)能捕获系统出现的异常,而系统异常往往令程序员头痛了,现
// 在系统一般都比较复杂,而且由很多人共同开发,一不小心就会导致一个
// 指针变量指向了其它非法区域,结果意外灾难不幸发生了。catch(…)为这种
// 潜在的隐患提供了一种有效的补救措施。
catch(…)
{
}
}

还有,特别是VC程序员为了使开发的系统有更好的可靠性,往往在应用程序的入口函数中(如MFC框架的开发环境下 CXXXApp::InitInstance())和工作线程的入口函数中加上一个顶层的trycatch块,并且使用catch(…)来捕获一切所有的 异常,如下:

BOOL CXXXApp::InitInstance()
{
if (!AfxSocketInit())
{
AfxMessageBox(IDP_SOCKETS_INIT_FAILED);
return FALSE;
}

AfxEnableControlContainer();

// Standard initialization
// If you are not using these features and wish to reduce the size
// of your final executable, you should remove from the following
// the specific initialization routines you do not need.

#ifdef _AFXDLL
Enable3dControls(); // Call this when using MFC in a shared DLL
#else
Enable3dControlsStatic(); // Call this when linking to MFC statically
#endif


// 注意这里有一个顶层的trycatch块,并且使用catch(…)来捕获一切所有的异常
try
{
CXXXDlg dlg;
m_pMainWnd = &dlg;
int nResponse = dlg.DoModal();
if (nResponse == IDOK)
{
// TODO: Place code here to handle when the dialog is
// dismissed with OK
}
else if (nResponse == IDCANCEL)
{
// TODO: Place code here to handle when the dialog is
// dismissed with Cancel
}
}
catch(…)
{
// dump出系统的一些重要信息,并通知管理员查找出现意外异常的原因。
// 同时想办法恢复系统,例如说重新启动应用程序等
}

// Since the dialog has been closed, return FALSE so that we exit the
// application, rather than start the application's message pump.
return FALSE;
}

   通过上面的例程和分析可以得出,由于catch(…)能够捕获所有数据类型的异常对象,所以在恰当的地方使用catch(…)确实可以使软件系统有着更 好的可靠性。这确实是大家使用catch(…)这个东东最好的理由。但不要误会的是,在C++异常处理模型中,不只有catch(…)方法能够捕获几乎所 有类型的异常对象(也许有其它更好的方法,在下一篇文章中主人公阿愚带大家一同去探讨一下),可C++标准中为什么会想到定义这样一个catch(…) 呢?有过java或C#编程开发经验的程序员会发现,在它们的异常处理模型中,并没有这样类似的一种语法,可这里不得不再次强调的是,java中的异常处 理模型是C++中的异常处理模型的完善改进版,可它反而没有了catch(…),为何呢?还是先去看看下一章吧,“C++的异常处理和面向对象的紧密关系 ”。也许大家能找到一个似乎合理的原因。
分享到:
评论

相关推荐

    C++_Builder_Code_Tech_超全~~

    C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech C++_Builder_Code_Tech ...

    Exceptions-and-error-handling.rar_c++ 异常_多条catch_异常

    1. **多条catch语句**:在C++中,一个`try`块可以跟随多个`catch`块,每个`catch`块处理不同类型的异常。这样设计的好处在于,你可以针对不同类型的异常采取不同的处理策略,提高代码的可读性和可维护性。例如: ``...

    1c++_C++_

    标题 "1c++_C++_" 暗示我们讨论的主题是关于C++编程语言的知识。C++是一种广泛应用的面向对象的编程语言,由Bjarne Stroustrup于1979年在贝尔实验室开发,作为C语言的扩展。它在C的基础上增加了类、模板、异常处理等...

    C++_Primer(四)答案(完整版_0积分)C++_Primer(四)答案(完整版_0积分)C++_Primer(四)答案(完整版_0积分)C++_Primer(四)答案(完整版_0积分)

    异常处理是现代软件开发中的一个关键概念,C++ Primer 深入讲解了try-catch-finally语句的使用,以及如何在C++中设计健壮的异常处理机制。 ### 7. 文件输入/输出 对于任何实用的程序而言,读写文件的能力至关重要...

    C++_GUI_Programming_with_Qt_4_中文版

    C++_GUI_Programming_with_Qt_4_中文版C++_GUI_Programming_with_Qt_4_中文版C++_GUI_Programming_with_Qt_4_中文版C++_GUI_Programming_with_Qt_4_中文版

    exp_C++_DEMO_beginner_

    《C++初学者指南:基于"exp_C++_DEMO_beginner_"的探索》 C++,作为一种强大的、通用的编程语言,深受程序员的喜爱。它不仅具有高效的性能,还提供了丰富的库支持和面向对象的特性。对于初学者来说,理解和掌握C++...

    21天学通C++(第五版) vs2008程序打包 深入浅出MFC 浙大教材C++达内 C++ 华为C++培训资料 visual c++_MFC

    21天学通C++(第五版) vs2008程序打包 深入浅出MFC 浙大教材C++达内 C++ 华为C++培训资料 visual c++_MFC 资源内容: visual c++_MFC 达内 C++ C++课件.ppt c++源码 .rar Core C++ Programming_new.ppt Core C++ ...

    OrderSystem_film36v_C++_预约系统_预约c++程序_

    【OrderSystem_film36v_C++_预约系统_预约c++程序】是一个基于C++语言开发的机房预约系统,旨在实现高效、便捷的机房资源管理与预约功能。这个系统的核心是利用C++的强大功能,为用户提供友好的交互界面和稳定的后台...

    c++ try catch.doc

    本文将深入探讨C++中的错误捕获机制——特别是`try-catch`结构,并通过实际案例分析其在不同编译环境下的行为差异。 #### 二、`try-catch`基本原理 在C++中,`try-catch`结构是最常见的异常处理方式之一。其基本...

    C++异常处理技巧try/catch/throw/finally/exception

    通过合理地使用`try`、`catch`、`throw`和`exception`等关键字,开发者不仅能够编写更加可靠的代码,还能显著提升程序的可维护性和可读性。 #### 二、C++标准异常处理 ##### 1. 构造和析构中的异常抛出 在C++中,...

    codes_helbrea_C++_code_源码.zip

    5. **异常处理**:用于捕获和处理程序运行时的错误,通过try、catch和throw关键字进行。 6. **内存管理**:C++允许直接操作内存,包括动态内存分配(new、delete)和智能指针(如unique_ptr、shared_ptr等)。 7. ...

    ezyzip_C++_

    《ezyzip_C++_》是一本专门为编程爱好者和初学者设计的C++教程,旨在帮助读者深入理解和熟练掌握这门强大的编程语言。C++,由Bjarne Stroustrup在1983年创立,是C语言的增强版本,它结合了过程化编程、面向对象编程...

    异常处理try catch在vc中的用法.zip visual c++ try catch处理异常

    在Visual C++(简称VC)环境下,我们可以利用`try`、`catch`关键字来实现异常处理。下面我们将深入探讨这些知识点。 首先,`try`块是异常处理的起点,它包围了可能抛出异常的代码。如果在`try`块内发生了异常,程序...

    ex4_C++_

    【标题】"ex4_C++_" 指的是一个与C++编程语言相关的课程实验,具体为第四次实验。这通常意味着它包含了学生在学习C++过程中,针对某个特定主题或问题编写的代码示例,可能包括了课堂上讨论的概念、算法实现或是编程...

    数据结构 in c++(中文版)Fundamentals_of_Data_Structures_in_C++_by_Horowitz

    Fundamentals_of_Data_Structures_in_C++_by_Horowitz_1997 最经典的数据结构教材的中文版(很多学校用的都是英文影印版吧)

    c++中try catch的用法小结

    C++ 中 try catch 的用法小结 try catch 机制是 C++ 编程语言中的一种异常处理机制,它允许程序员捕捉和处理在程序执行过程中出现的异常情况。下面将详细介绍 C++ 中 try catch 的用法。 基本概念 在 C++ 中,try...

    cppreference-zh-20191207_cppreference_C++_zip_

    【cppreference-zh-20191207_cppreference_C++_zip_】这个压缩包文件主要包含了中文版的C++参考手册——cppreference-zh,版本为2019年12月7日更新。cppreference是一个非常权威且全面的C++编程语言在线文档,它详细...

    范磊c++_从新手到高手视频下载地址

    3. 异常处理:C++提供了异常处理机制,通过try、catch和throw关键字来捕获和处理运行时错误,提高程序的健壮性。 4. 标准模板库(STL):STL是C++的一个重要组成部分,包括容器(如vector、list、set、map等)、...

    wsc.rar_C++_C++ 编码规范_C++大作业_handle3ne_清华C++大作业

    5. **异常处理**:C++支持异常处理,通过try、catch和throw关键字来捕获和处理运行时错误。良好的异常处理能确保程序在遇到异常情况时能够优雅地失败。 6. **模板与泛型编程**:C++的模板机制允许创建泛型代码,...

Global site tag (gtag.js) - Google Analytics