




已阅读5页,还剩10页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
精品文档目录:一、 SEH的概念、特性二、 SEH的基本使用方法1、 结束异常程序(1)try块的自然退出与非自然退出(2)finally块的清理功能及对程序结构的影响(3)关键字_leave2、 异常处理程序(1)异常处理的基本流程(2)异常过滤器(3)全局展开(4)暂停全局展开3、 未处理异常(顶层异常处理)三、 SEH相关数据结构的介绍1、 EXCEPTION_POINTER结构2、 EXCEPTION_RECORD结构3、 EXCEPTOIN_REGISTRATION结构4、异常处理链结构图四、 VC+编译器级SEH的具体实现1、 VC扩展异常帧2、 VC异常帧堆栈布局3、 两个实例程序:显示异常帧信息4、 实例分析及特性介绍5、 VC中的定层异常处理6、 VC搜索异常处理程序流程五、参考资料- 前言: 对于这片文章应该是我写的最认真的一篇了,断断续续地写了将近一个月,从最初的复习性的回顾,收集更多资料,反复地整理思路,查阅Windows源码中的相关的源码,设计文章的整体框架,到每一个部分的详细设计,包括流程图的设计。每一个过程都进展的并不顺利,由于时间的关系每星期只能有23次的大段时间的在电脑面前,所以思路一直在被打断,整理文章写的并不流畅。有些思路或许已经不知道遗忘在哪个角落,但我也尽量把相关方面的知识点都讲到。本文的重点在于SEH原理方面的介绍,从单纯的使用角度来看,比较简单,但如果适当的使用SEH机制,这在很大程度上与使用者对SEH的理解程度有很大关系,因而对于具体的实现,文中只介绍重点及需要注意的地方,具体使用方法可参见参考文献1。一、 SEH的概念、特性 SEH,结构化异常处理,是作为一种系统机制引入到操作系统中的,本身与语言无关。在我们自己的程序中使用SEH可以让我们集中精力开发关键功能,而把程序中所可能出现的异常进行统一的处理,使程序显得更加简洁且增加可读性。 当在程序中使用SEH时,就变成编译器相关的。其所造成的负担主要由编译程序来承担,例如编译程序会产生一些表(table)来支持SEH的数据结构,还会提供回调函数。二、 SEH的基本使用方法1、 结束异常程序 一个结束异常程序能确保调用和执行一个代码块,对应与具体的实现,结束处理程序的结构如下所示: _try / 受保护的代码 _finally / 结束处理程序 (1)try块的自然退出与非自然退出 try块可能会因为return,goto,异常等非自然退出,也可能会因为成功执行而自然退出。但不论try块是如何退出的,finally块的内容都会被执行。 请看下面两个程序: 通过使用结束处理程序,可以避免return语句的过早执行。当retrun 试图退出try块时,编译程序要保证finally块中的代码首先被执行。这事实上就是一个局部展开的过程,当从try块的过早退出强制控制转移到finally块时,都将引起局部展开。(2)finally块的清理功能及对程序结构的影响 写过软件的朋友一般都有这样一个影响:在编码的过程中需要加入需要检测,检测功能是否成功执行,若成功的话执行这个,不成功的话需要作一些额外的清理工作,例如释放内存,关闭句柄等。如果检测不是很多的话,倒没什么影响;但若又许多检测,且软件中的逻辑关系比较复杂时,往往需要化很大精力来实现繁芜的检测判断。结果就会使程序看起来结构比较复杂,大大降低程序的可读性,而且程序的体积也不断增大。 事实上可以用SEH 来解决,把一些相关函数的清理代码都放在finally块,只需要在其中加一些适当的判断,不需要回到每个可能失败的地方添加清理代码。下面的FunSampleA函数是一个常规的函数,FunSampleB引入了SEH结束处理程序机制: 这两个函数的功能是一样的。可以看到在FunSampleA中的清理函数(CloseHandle)到处都是,而在FunSampleB中的清理函数则全部集中在finally块,如果在阅读代码时只需看try块的内容即可了解程序流程。这两个函数本身都很小,可以细细体会下这两个函数的区别。(3)关键字_leave 在try块中使用_leave关键字会使程序跳转到try块的结尾,从而自然的进入finally块。 对于上例中的FunSampleB,try块中的3个return完全可以用_leave来替换。两者的区别是用return会引起try过早退出系统会进行局部展开而增加系统开销,若使用_leave就会自然退出try块,开销就小的多。 但有一种情况下必须使用_leave而不能使用return,即当finally块后还需要执行一定的功能,如下所示: 2、异常处理程序异常处理程序能在程序发生异常时进行相应的处理,对应与具体的实现,异常处理程序的结构如下所示: _try / 受保护的代码 _except ( /*异常过滤器exception filter*/ ) / 异常处理程序exception handler (1)异常处理的基本流程(注:此流程图来源于参考资料1) (2)异常过滤器异常过滤器只有三个可能的值(定义在Windows的Excpt.h中):EXCEPTION_EXECUTE_HANDLEREXCEPTION_CONTINUE_SERCHEXCEPTION_CONTINUE_EXECUTION下面是两种基本的使用方法:方式一:直接使用过滤器的三个返回值之一_try _except ( EXCEPTION_EXECUTE_HANDLER ) 方式二:自定义过滤器_try _except ( MyFilter( GetExceptionCode() ) ) LONG MyFilter ( DWORD dwExceptionCode ) if ( dwExceptionCode = EXCEPTION_ACCESS_VIOLATION ) return EXCEPTION_EXECUTE_HANDLER ; else return EXCEPTION_CONTINUE_SEARCH ;(3)全局展开 首先来看一下全局展开的基本流程(此流程图来源于参考资料1): 接下来看一个全局展开的实例(源码GlobalUnwindSample.cpp):代码:#include #include static unsigned int nStep = 1 ;void Function_B () int x, y = 0 ; _try x = 5 / y ; / 引发异常 _finally cout Step nStep+ : 执行Function_B的finally块的内容 endl ; void Function_A ( ) _try Function_B () ; _finally cout Step nStep+ : 执行Function_A的finally块的内容 endl ; long MyExcepteFilter ( ) cout Step nStep+ : 执行main的异常过滤器 endl ; return EXCEPTION_EXECUTE_HANDLER ;int main () _try Function_A () ; _except ( MyExcepteFilter() ) cout Step nStep+ : 执行main的except块的内容 handler, pVCExcRec-prev, pVCExcRec-scopetable ); PSCOPETABLE pScopeTableEntry = pVCExcRec-scopetable; for ( int i = 0; i trylevel; i+ ) printf( scopetable%u PrevTryLevel: %08X filter: %08X _except: %08Xn, i, pScopeTableEntry-previousTryLevel, pScopeTableEntry-lpfnFilter, pScopeTableEntry-lpfnHandler ); pScopeTableEntry+; printf( n ); /显示异常帧信息/void SEHShowExcptFrames( ) PVC_EXCEPTION_REGISTRATION pVCExcRec; / 取得异常帧链首地址,保存在pVCExcRec _asm mov eax, FS:0 _asm mov pVCExcRec, EAX / 遍历异常帧链 while ( (unsigned)pVCExcRec != 0xFFFFFFFF ) SEHShowScopeTable( pVCExcRec ); pVCExcRec = (PVC_EXCEPTION_REGISTRATION)(pVCExcRec-prev); 嵌套异常结构示例一: 输出结果: 嵌套异常结构示例二: 输出结果: 4、实例分析及特性介绍(比较分析,注意结构示例与结果之间的联系) 1 上面这个两个结果中各有4个异常帧。 2 他们的后两条异常帧都是一致的,变化的只是前两个异常帧 3 都是两重函数调用,mainFunction,每一重调用都以为着增加一个异常帧,其中的第一个帧代表最内层异常块,即Function函数的异常帧(相对于显示异常帧信息的函数来说),第二帧代表main函数的异常帧。 4 scopetable域的项数与该函数中的异常块数目相关,示例一中的Function函数中有三个嵌套的异常块,因而第一帧scopetable中有三项;示例二中的Function函数中有四个异常块,因而第二帧scoptable中有四项。 5 首先来看示例一,Function函数中的三个异常块是嵌套的,所以在scopetable2中的PrevTryLevel=1,表示当前异常块在帧中的索引为2,与它相邻的外层异常块的索引为1,这个异常块的异常过滤器地址为filter:00401204,而异常处理回调函数的地址为_except: 00401207;因而在示例一的第一帧scopetable中的三项形成一个链表(这也比较好理解:三个异常块嵌套)。再来看示例二,还是看第一帧,自己分析下,是否形成了两个链表。到这里,对于scoptable中的PrevTryLevel的意义应该清楚了。如果对于这部分还不是很理解的话可以通过增加或修改程序中异常块、或者再增加几层函数调用,然后观察分析输出结果。 6 不知道大家有没有发现EXCEPTION_REGISTRATION结构中有个handler,而在SCOPETABLE中也有个lpfnHandler(即上面例子输出结果中的_except项),这两个都是异常处理回调函数的地址,它们之间到底有什么联系和区别?示例中前三帧的handler地址都是004014B0,why? 这个也不难回答,scopetable中的lpfnHandler是用于用户自定义的异常处理函数,而EXCEPTION_REGISTRATION的handler是用于系统默认的异常处理程序,即通常说的顶层异常处理回调函数。 7 分析到这里,我只分析了4帧中的两帧,至于后两帧的作用,这一点将会在后文中讲解。5、VC中的顶层异常处理 是否还记得在介绍第二部分的未处理异常时提到的BaseProcessStart,BaseThreadStart这两个函数。在启动主/次线程时,系统会把整个线程置于一个异常块中,即顶层异常处理体。在上文显示异常帧的两个例子中的第四帧正是来源于此。 在默认情况下,VC程序的入口点由运行时库函数mainCRTStartup和WinMainainCRTStartup来实现(如果对这方面不熟悉的话,请参见参考资料5),其中又一次对将要调用的用户程序入口main,WinMain(默认情况)置于异常处理体中,在上文显示异常帧的两个例子中的第三帧来自这里。对于这两个入口函数,CRT中有相应的源码,大致结构经简化如下所示: 现在可以来总结下,VC编译器中默认情况(即我们的程序不显示的引入SEH)下的异常块结构示例: 其中的BaseProcessStart,WinMainCRTStartup这两个函数并非固定,需要是视具体情况而定,这个只是用做示例。 到这里,对于上小节中的遗留下来的关于第三和第四个异常帧的问题已经解决了。6、VC搜索异常处理程序流程 在介绍第三部分SEH相关数据结构时,已经介绍了异常处理链的结构,但针对VC中的异常处理,异常帧的扩展包括scopetable结构的引入使得支持嵌套的异常结构,因而当出现异常时寻找相应的异常处理程序的过程相对来说变的更加复杂。VC搜索异常处理程序流程如下所示: 【参考文献】1. Windows核心编程 Jeffrey Richter著 机械工业出版社2. A Crash Course on the Depths of Win 32® Structured Exception Handling/msj/0197
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 出租车安全夏天培训内容课件
- 急救知识培训考试试卷及答案
- 出海作业安全培训课件
- 2025合同范本合同法规定
- 《市场营销学》作业考核试题及答案
- 2025标准汽车租赁合同样本
- 道法考试试题分析模板及答案
- 2025《合同风险评估与控制》
- 冲焊部员工安全培训课件
- 高中化学必修第三章第四节基本营养物质
- 生物质颗粒购销合同
- 第01讲 意象、画面与意境 练习 中考语文复习
- 2025年湖南水利水电职业技术学院单招职业技能测试题库附答案
- 第四章药品调剂质量控制第二节用药错误管理课件
- 幼儿园酸奶牛奶采购合同
- 高中化学拔尖创新人才早期培养路径
- 2025年四川宽窄实业有限责任公司招聘笔试参考题库含答案解析
- 《中药提取物生产技术》课件-中药常用的粉碎方法
- Unit 1 完形填空训练8篇-2023-2024学年英语八年级上册单元冲刺满分题型训练(人教版)
- DB32/T 1086-2022 高速公路建设项目档案管理规范(修订)
- 教师资格证《教育知识与能力》中学-必背知识点
评论
0/150
提交评论