来源:http://bbs.ustc.edu.cn/cgi-bin/bbsgcon?bn=CPlusPlus&fn=G431EF019&num=224
发信人: cppmm (0123单身情鸽|追风坠月不知东西), 信区: CPlusPlus
标 题: 异常处理方法,同步VS异步zz
发信站: 瀚海星云 (2005年09月07日21:50:18 星期三), 站内信件 WWWPOST
VC默认情况下是同步异常捕捉,所以只有throw才算是正式异常,而
对内存访问出错和除零则不保证正能正确解栈(可能导致栈上的对象
未被析构)
1. Windows SEH 与 C++ Exception
1) Windows SEH 结构化异常
结构化异常是Windows操作系统提供的与语言无关的异常处理机制, SHE使用
Win32API中的RaiseException()函数来抛出异常,在VC中使用关键字__try和关键
字__except来捕获,并用宏函数GetExceptionCode和GetExceptionInfo来获取捕
获的异常由什么原因产生,和产生异常时环境状态。__finally关键字保证无论是否发生
异常,finally代码段都会被执行。
SHE使用示例代码
int ECode;
__try
{
__try
{
RaiseException(1, // 抛出异常码为1的SEH异常
0,
0, NULL); // 没有参数
}
__finally
{
printf(2 ); // 不管是否有异常,必定会执行的代码
}
}
__except (ECode=GetExceptionCode())
{
printf(发生异常,Code=%d/n,ECode); // 捕获异常后执行的代码 ;
}
输出结果:
2发生异常,Code=1
2) C++Exception
C++标准也提供了一种异常处理机制,通过使用try,catch,throw关键字来表达,在
C++异常可以通过throw函数来抛出简单变量,复杂变量与异常对象,与Windows异
常相比,异常对象可以给开发者提供更多的信息。
try
{
//正常代码
。。。
throw CExcetion();
。。。
}
catch (CException* e)
{
//处理异常代码
}
3) SEH 到 C++异常的转换
在同一个程序中,如果使用WIN32API它会抛出SHE,使用C++库函数,它们又会抛出
C++异常,Win32API和C++函数混和使用时如果使用两种异常捕获机制时,使用起
来会影响程序的可读性,因此C++运行库提供了_set_se_translator函数,在SHE异
常发生时通过回调方式来转换SEH异常为C++异常。在此提供一个转换的宏来实现转
换。
转换宏的代码:
#define INSTALL_SEHCONVERT() ExceptionConvert ecExceptionConvert
class SEHException
{
private:
unsigned int nSE;
public:
SEHException() {}
SEHException( unsigned int n ) : nSE( n ) {}
~SEHException() {}
unsigned int getSeNumber() { return nSE; }
};
class ExceptionConvert
{
public:
ExceptionConvert(){OldFanc = _set_se_translator(trans_func); }
~ExceptionConvert(){_set_se_translator(OldFanc); }
private:
static void trans_func( unsigned int u,
EXCEPTION_POINTERS* pExp )
{
throw SEHException(u);
}
_se_translator_function OldFanc;
};
使用上面INSTALL_SEHCONVERT宏后就可以使用如下代码来捕获SHE异常了
INSTALL_SEHCONVERT();
Try
{
…
}
catch(SEHException &seh){
…
}
2. 同步异常与异步异常
1) VC的C++ Exception 采用两种模式捕获异常:同步模式和异步模式。VC的工程的
调试版本缺省使用异步模式,工程的发布版本缺省使用同步模式。在同步模式下,VC的
编译器假定代码中只有在显示使用throw和调用函数的时候才会引发异常,因此,在同
步模式下,VC编译出的代码比较小,但在这种模式下,try-catch对不能捕获内存访问
异常与算术除零异常等。在异步模式下,VC的编译器为try块内的每一条语句生成异常
捕获代码,在这种情况下,他能够捕获全部的异常,还能保证栈上对象在解栈中正确释
放。为了要在发行版本中也能够捕获全部异常就需要打开异步模式,但代价是程序编译
出代码变大,运行速度变慢。
2)编译选项:
同步模式的编译选项为/EHs或者/GX(等同于/EHsc)
异步模式的编译选项为/EHa
3. 多线程下的异常捕获
在创建线程并运行线程的函数中把创建线程的代码放在try块中并不会捕获到线程函数中
发生的异常,线程函数中发生的异常只能在线程函数中捕获。并且每一个线程都需要自
己的SHE转换宏。转换宏可以放在线程函数的开始部分
4.参考MSDN库