SEH:__try__leave__finally__exception;try_catch_throw

来源:互联网 发布:淘宝新手代理 编辑:程序博客网 时间:2024/05/14 03:22

__try

{

// condition test

__leave;

//another conditon test

__leave;

//so on

}

__finally

{

//do some clean up job here

}


#include "stdafx.h"
#include <windows.h>
#include <excpt.h>

int filter(unsigned int code, struct _EXCEPTION_POINTERS *ep)

if (code == EXCEPTION_ACCESS_VIOLATION)
{  
return EXCEPTION_EXECUTE_HANDLER;//捕获到异常  

else
{  
return EXCEPTION_CONTINUE_SEARCH;  

}


int main(int argc, char* argv[])
{
int* p = NULL;   // pointer to NULL

int szBuf[] = {1};
p = szBuf;

__try
{  
__try
{   
puts("try"); 
printf("%d\n", *p);//puts(p);

   // __leave;
// printf("aaa");
//*p = 13;    // causes an access violation exception;

}
__finally
{   
puts(AbnormalTermination() ? "abnormal" : "normal");   
}

}
__except(filter(GetExceptionCode(), GetExceptionInformation()))
{  
puts("except");

return 0;
}


try-except用法

try except是windows 系统独有的异常处理模型,windows的异常处理模式,称为SEH( structured exception handling ),SEH的异常处理模型主要由try-except语句来完成,与标准的try catch相似。与C++异常处理模型使用catch关键字来定义异常处理模块,而SEH是采用__except关键字来定义。并且,catch关键字后面往往好像接受一个函数参数一样,可以是各种类型的异常数据对象;但是__except关键字则不同,它后面跟的却是一个表达式.

我们知道,函数调用也是一个表达式。

我们来看下面这个例子,这个例子是用来处理栈溢出的异常。

long WINAPI FilterFunc(DWORD dwExceptionCode)

{

return (dwExceptionCode == STATUS_STACK_OVERFLOW) ? EXCEPTION_EXECUTE_HANDLER : EXCEPTION_CONTINUE_SEARCH;

}

UINT WINAPI ThreadFunc(LPVOID param)

{

__try

{

   // guarded code

}

__except (FilterFunc(GetExceptionCode()))

{

// 如果是栈溢出,进行处理。

}

   return TRUEt;

}

except参数的值有以下三种:
EXCEPTION_CONTINUE_EXECUTION (–1) 异常被忽略,控制流将在异常出现的点之后,继续恢复运行。
EXCEPTION_CONTINUE_SEARCH (0) 异常不被识别,也即当前的这个__except模块不是这个异常错误所对应的正确的异常处理模块。系统将继续到上一try-except域中继续查找一个恰当的__except模块。

EXCEPTION_EXECUTE_HANDLER (1) 异常已经被识别,控制流将进入到__except模块中运行异常处理代码

try-except的关键是如何在__except模块中获得异常错误的相关信息.

Windows提供了两个API函数来获取异常信息:

LPEXCEPTION_POINTERS GetExceptionInformation(VOID); //取得异常相关信息

DWORD GetExceptionCode(VOID); // 取得异常编号

GetExceptionCode()返回异常编号,而GetExceptionInformation()返回更丰富的信息,EXCEPTION_POINTERS结构如下,

typedef struct _EXCEPTION_POINTERS { // exp

PEXCEPTION_RECORD ExceptionRecord;

PCONTEXT ContextRecord;

} EXCEPTION_POINTERS;

其中EXCEPTION_RECORD类型,它记录了一些与异常相关的信息;而CONTEXT数据结构体中记录了异常发生时,线程当时的上下文环境,主要包括寄存器的值。

有了这些信息,__except模块便可以对异常错误进行很好的分类和恢复处理,通常我们需要一个过滤函数来辅助。一般称为是filterfunction.过滤函数只过滤需要处

理的异常。

int exception_access_violation_filter(LPEXCEPTION_POINTERS p_exinfo)

{

   if(p_exinfo->ExceptionRecord->ExceptionCode == EXCEPTION_ACCESS_VIOLATION)

   {

   messagebox("access vialation exceptionn");

   return EXCEPTION_EXECUTE_HANDLER ; //告诉except处理这个异常

   }

   else return EXCEPTION_CONTINUE_SEARCH; //不告诉except处理这个异常

}

int exception_int_divide_by_zero_filter(LPEXCEPTION_POINTERS p_exinfo)

{

   if(p_exinfo->ExceptionRecord->ExceptionCode == EXCEPTION_INT_DIVIDE_BY_ZERO)

   {

   return EXCEPTION_EXECUTE_HANDLER; //告诉except处理这个异常

   }

   else return EXCEPTION_CONTINUE_SEARCH; //不告诉except处理这个异常

}

于是,你可以这样写这段异常处理代码:

__try

{

   // guarded code

}

__except(exception_access_violation_filter(GetExceptionInformation()))

{

//

}

__try

{

   // guarded code

}

__exceptexception_int_divide_by_zero_filter(GetExceptionInformation()))

{

//exception handling

}

SEH异常处理模型中,也可以抛出一个异常。对应的WindowsAPI函数是RaiseException,

VOID RaiseException(

DWORD dwExceptionCode, // 异常的编号

DWORD dwExceptionFlags, // 异常标记

DWORD nNumberOfArguments, // 参数个数

CONST DWORD *lpArguments // 参数数组首地址

);

通常,后三个参数基本不用

SEH异常处理还有try-finally.类似于java里的try-catch-finally.但是SEH的try只能和except和finally两者之间的一个搭配,不能有try-except-finnaly.

C++异常模型用try-catch语法定义,而SEH异常模型则用try-except语法,与C++异常模型相似,try-except也支持多层的try-except嵌套。

try-except模型中,一个try块只能是有一个except块;而C++异常模型中,一个try块可以有多个catch块。

C++异常模型是按照异常对象的类型来进行匹配查找的;而try-except模型则不同,它通过一个表达式的值来进行判断.

__except关键字后面跟的表达式,它可以是各种类型的表达式,例如,它可以是一个函数调用,或是一个条件表达式,或是一个逗号表达式,或干脆就是一个整

型常量等等。最常用的是一个函数表达式,并且通过利用GetExceptionCode()或GetExceptionInformation ()函数来获取当前的异常错误信息,便于程序员有效控制异常

错误的分类处理。

SEH异常处理模型中,异常通过RaiseException()函数抛出。RaiseException()函数的作用类似于C++异常模型中的throw。

关于SEH异常处理更详细的资料,你可以去看windows via c/c++这本书,中文译名是windows核心编程。不过还是建议你看英文原版,翻译的版本质量不高。

try
{

}
catch(XXException & e)
{

}
catch(YYException & e)
{

}
catch(...)
{
     e.GetMessage();
     throw(e);
}

原创粉丝点击