异常处理方法
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库