如何捕获access violation异常

文章目录

  1. access violation的由来
  2. access violation的实例
  3. Win32 exception
  4. SEH异常与C++标准异常
  5. 捕获方法

 

1.access violation的由来

access violation,字面意思就是试图访问不可访问的内存。比如给一个标记为NULL的指针赋值。

通常这个异常在C++中会引起程序崩溃,并提示“异常代码:c0000005”。这个异常属于Windows操作系统定义的硬件异常,

参见MSDN关于硬件异常的说明:http://msdn.microsoft.com/zh-cn/Library/w49wew4f.aspx。

2. access violation的实例

 示例1:对空指针进行赋值

 

char * p=NULL; 
p="abcd"; 


示例2:修改只读内存

char * sss="abcd";
sss[0]='A'; //sss指向只读区域,不可改写

3.Win32 exception

说到win32 exception,就要了解Windows提供的SEH(Structured Exception Handling)服务。

简单的说,SEH服务是Windows作为操作系统,为了适配各种CPU和其他底层软件,向上层应用程序

提供的一种统一的异常服务。

SEH的异常包括了软件异常和硬件异常。

像access violation ,除0这些就属于硬件异常。

关于windows操作系统异常的定义,可参见microsoft sdk中的ntstatus.h头文件,比如

G:\Program Files\Microsoft SDKs\Windows\v7.0A\Include\ntstatus.h
更多关于Win32异常的理解参见CodeProject上的文章:
http://www.codeproject.com/Articles/82701/Win32-Exceptions-OS-Level-Point-of-View
4.SEH异常与C++标准异常

  • C++标准异常在Windows上的实现依赖SEH异常。
  • SEH异常是通过Windwos API ,RaiseException抛出的,每个不同的异常,规定了不同的异常代码。比如access violation,

异常代码是c0000005。

  • C++标准异常在底层也是通过RaiseException抛出(throw内部调用RaiseException)的,但是他的异常代码都是相同的,都是0xE06D7363,对应的“.msc”的ASCII码。当一个SEH异常抛出时,如果它的异常代码是0xE06D7363,就会被认定为C++标准异常,程序转交给C++异常处理机制,C++语言在这个之后,依据异常的类型,再进行C++层次的异常处理。

总结:

C++异常是底层是通过SEH异常服务实现的,是SEH异常的一种。C++语言的标准异常处理机制是在收到SEH抛出的

异常之后,在C++语言层实现的异常处理服务。SEH异常是通过异常代码来标记异常的,C++标准异常是通过异常数据类型来区分异常的。

5.捕获方法

(1)标准C++捕获SEH异常

try()catch()是C++提供的异常处理机制,不能捕获SEH异常,因为只有异常代码0xE06D7363为的SEH异常,才会转交给

C++标准异常处理。但是在VC编译器下,依据版本的不同会有变化。


  • VC++6.0 try catch默认可以捕获SEH异常
  • VC++8.0 默认不可以,开启/EHa编译命令后可以
(2)使用VC捕获SEH异常

__try()__except()是微软提供用于捕获SEH异常的机制,在VC编译器中实现。详细可以参见msdn关于__try()__except()的说明,这样,我们就可以抓取access violation这样的异常了,并且可以使用GetExceptionCode(),GetExceptionInformation()等这些WinAPI,记录错误的异常代码,调用堆栈。

(3)更多关于在托管代码和非托管代码中捕获SEH异常的内容可参见

MSDN:Exception Handling Differences

CodeProject:Effective Exception Handling in Visual C++





你可能感兴趣的:(access,异常处理,操作系统)