C++编译器怎么实现异常处理1

原创 2002年09月17日 09:28:00

C++编译器怎么实现异常处理

<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

 

对于VC++实现异常处理的深入探讨

 

导论

 

相比较其他传统的语言,C++的一个变革的特征是支持异常处理。相对于传统语言的不清楚容易错误的错误处理机制,C++的异常处理是一个非常好的替代。在正常的代码和错误处理代码之间清楚的分割使得程序非常整洁和宜于维护。本文讨论编译器怎么实现异常处理。假设读者熟悉异常处理的语法。 本文包含一个异常处理的VC++的库来替代VC++的异常处理,使用这个函数:

 

install_my_handler();

 

在这以后,程序中发生的任何异常(包含抛出异常到 stack unwinding,调用catch块和继续执行)都使用我自己的异常处理库。

 

译者注:当异常出现时,正常的执行流被中断,异常处理机制开始在当前范围寻找匹配的处理函数。如果找不到,把当前函数从栈中弹出,在调用者中继续寻找。这个过程称为stack unwinding

 

像其他C++特征一样,C++的标准并没有指定异常处理的实现机制。这使得C++实现者可以使用任何实现机制。我将讲述VC++怎么实现的。VC++把异常处理置于SHEStructured exception hangling)的上面。SHEwindows操作系统提供的结构化的异常处理。

 

SHE导论

 

在本讨论中,我将考虑那些显式的异常。例如被0除,空指针访问等。当异常出现,中断会产生,控制被转到OS OS调用异常处理,检查从异常发出的函数开始的函数调用顺序,执行stack unwinding和控制转移。我们可以开发自己的异常处理函数,在OS中注册。OS就会在异常事件时调用它们。

 

Windows定义了一个特别的结构用来注册:

 

EXCEPTION_REGISTRATION:

struct EXCEPTION_REGISTRATION

{

   EXCEPTION_REGISTRATION *prev;

   DWORD handler;

};

 

要注册自己的异常处理函数,创建这个结构并将它的地址保存在段(由FS寄存器指向)的0偏移处。如下面的伪汇编指令:

 

mov FS:[0], exc_regp

 

结构中的prev字段表示EXCEPTION_REGISTRATION链表。当我们注册了这个EXCEPTION_REGISTRATION结构,我们使用这个prev字段保存以前注册的结构的地址。

 

关于异常回调函数,windows要求异常处理的信号,定义在excp.h文件中:

 

EXCEPTION_DISPOSITION (*handler)(

    _EXCEPTION_RECORD *ExcRecord,

    void * EstablisherFrame,

    _CONTEXT *ContextRecord,

    void * DispatcherContext);

 

现在你可以忽略所有的参数和返回类型。下面的程序在OS中注册了一个异常处理句柄并将产生一个被0除的异常。这个异常被抓到并将打印一个消息:

 

#include <iostream>

#include <windows.h>

 

using std::cout;

using std::endl;

 

 

struct EXCEPTION_REGISTRATION

{

   EXCEPTION_REGISTRATION *prev;

   DWORD handler;

};

 

 

EXCEPTION_DISPOSITION myHandler(

    _EXCEPTION_RECORD *ExcRecord,

    void * EstablisherFrame,

    _CONTEXT *ContextRecord,

    void * DispatcherContext)

{

         cout << "In the exception handler" << endl;

         cout << "Just a demo. exiting..." << endl;

         exit(0);

         return ExceptionContinueExecution; //will not reach here

}

 

int g_div = 0;

 

void bar()

{

         //initialize EXCEPTION_REGISTRATION structure

         EXCEPTION_REGISTRATION reg, *preg = &reg;

         reg.handler = (DWORD)myHandler;

        

         //get the current head of the exception handling chain      

         DWORD prev;

         _asm

         {

                 mov EAX, FS:[0]

                 mov prev, EAX

         }

         reg.prev = (EXCEPTION_REGISTRATION*) prev;

        

         //register it!

         _asm

         {

                 mov EAX, preg

                 mov FS:[0], EAX

         }

 

         //generate the exception

         int j = 10 / g_div;  //Exception. Divide by 0.

}

 

int main()

{

         bar();

         return 0;

}

 

/*-------output-------------------

In the exception handler

Just a demo. exiting...

---------------------------------*/

 

 

注意:windows严格地定义了一个规则:EXCEPTION_REGISTRATION结构应该在栈内,并且要在以前的代码的低的内存地址。规则不满足,windows将中止程序。

C++编译器怎么实现异常处理

对于VC++实现异常处理的深入探讨   导论   相比较其他传统的语言,C++的一个变革的特征是支持异常处理。相对于传统语言的不清楚容易错误的错误处理机制,C++的异常处理是一个非常好的替代。在正常的...
  • iiprogram
  • iiprogram
  • 2008年03月18日 21:36
  • 628

C++编译器怎么实现异常处理3

C++和异常再回头来说我们在第一节里说到的 EXCEPTION_REGISTRATION结构,这个结构是用来注册操作系统的异常回调函数的,当异常发生时,该函数将被调用。 VC++扩展了异常回调函数得语...
  • ancienttale
  • ancienttale
  • 2002年11月17日 11:51
  • 836

C++编译器怎么实现异常处理2

看了C++编译器怎么实现异常处理1    sdssly(翻译)http://www.csdn.net/Develop/article/15%5C15051.shtm没有下文,于是自己去看原文,也翻译了...
  • ancienttale
  • ancienttale
  • 2002年11月15日 14:01
  • 943

C++编译器怎么实现异常处理4

C++和异常2图 5 显示了函数信息(funinfo)结构的内容。请注意结构使用的名字可能和VC++编译器使用的实际名字不一样,而且我在图中只显示了有关的成员,结构中的unwind table成员我将...
  • ancienttale
  • ancienttale
  • 2002年11月21日 10:06
  • 1031

自制编译器---c++实现词法分析器

词法单元词法解析器在编译器中的作用,是将输入流解析为一种能够被语法解析器使用和管理的格式。他将输入文本分割,打标签,也就是用一些数值来指代一系列相应的字符串。 例如:int a,b,c; a=34;...
  • taoyanqi8932
  • taoyanqi8932
  • 2016年06月25日 21:45
  • 2093

C++编译器如何实现异常处理

C++编译器如何实现异常处理 原著:Vishal Kochhar 翻译:局部变量 原文出处:How a C++ compiler implements exception handlin...
  • keroro520
  • keroro520
  • 2013年12月21日 11:52
  • 529

晒晒C++:虚函数的真相(VC编译器如何实现“virtual ”规则)

可到 CSDN 下载中心下载全文 http://download.csdn.net/detail/Dreamcode/201005 ( 1 )virtual 虚函数 先看一段简单代码:Code Seg...
  • Dreamcode
  • Dreamcode
  • 2011年12月14日 22:39
  • 14179

c++类的编译器实现方式描述

貌似有些同学还不太明白这个,我试着用c代码描述c++类相关的一些实现方式。   设类abc,bc继承于a,都有个虚函数f(),析构函数为虚 c++代码 //---------A  struc...
  • Kevin_qing
  • Kevin_qing
  • 2011年08月24日 15:04
  • 1097

深入理解C++中的异常处理机制

深入理解C++中的异常处理机制异常处理增强错误恢复能力是提高代码健壮性的最有力的途径之一,C语言中采用的错误处理方法被认为是紧耦合的,函数的使用者必须在非常靠近函数调用的地方编写错误处理代码,这样会使...
  • zhangyifei216
  • zhangyifei216
  • 2015年12月26日 21:29
  • 3072

C++入门(1):程序、编译器和操作系统

程序、编译器和操作系统参考书目《C++ primer》《编译原理基础》 在开始任意一门语言的学习之前,很多人习惯于使用一个“软件”(如visual studio 2010),这个软件的功能是你可以...
  • huiyuanliyan
  • huiyuanliyan
  • 2016年06月27日 11:41
  • 2143
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C++编译器怎么实现异常处理1
举报原因:
原因补充:

(最多只允许输入30个字)