一、 让程序在崩溃时体面的退出之Unhandled Exception :
程序是由代码编译出来的,而代码是由人写的。人非圣贤,孰能无过。所以由人写的代码有缺陷是很正常的。当然很多异常都在开发阶段被考虑到而添加了处理代码,或者用try/catch对可能出现异常的地方进行额外的照顾。可是,还是会有一些无法预料的异常(Unhandled Exception)在程序运行的时候出现。这些异常很多时候都会招致程序的崩溃。那么有没有什么方法可以让程序在崩溃的时候体面的退出呢?答案是肯定的。可以用Windows API中的SetUnhandledExceptionFilter来设置一个回调函数来处理这些无法预料的异常。
要想使用SetUnhandledExceptionFilter,必须#include <Windows.h>,这个函数的声明如下:
具体的参数和返回值的解释可以查找MSDN,有很详细的说明。下面就用具体的代码来说明怎样使用这个Windows API来使程序在崩溃的时候体面的退出。代码里面有详细的注释来帮助理解。
用VC创建一个名为Test的控制台程序,添加下面的代码。到工程的Debug目录下找到编译好的Test.exe,双击运行。
#include "stdafx.h"
- // 处理Unhandled Exception的回调函数
- LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException)
- {
- // 在这里添加处理程序崩溃情况的代码 ,例如.cpi图的保存
- // 现在很多软件都是弹出一个发送错误报告的对话框
- // 这里以弹出一个错误对话框并退出程序为例子
- FatalAppExit(-1, _T("*** Unhandled Exception! ***"));
- return EXCEPTION_EXECUTE_HANDLER;
- }
- int _tmain(int argc, _TCHAR* argv[])
- {
- // 设置处理Unhandled Exception的回调函数
- SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);
- // 除零,人为的使程序崩溃
- int i = 13;
- int j = 0;
- int k = i / j;
- return 0;
- }
编译上面的代码,到工程的Debug目录下找到编译好的Test.exe,双击运行,就会出现下面这样的对话框,点击OK按钮后,程序正常退出。
发现:空指针,整数除0,boost::any_cast<>()错类型转换等均可捕获到,数组越界访问捕获失改。出现过蓝屏。
二、让程序在崩溃时体面的退出之Dump文件:
在我的那篇《让程序在崩溃时体面的退出之CallStack》中提供了一个在程序崩溃时得到CallStack的方法。可是要想得到CallStack,必须有pdb文件的支持。但是一般情况下,发布出去的程序都是Release版本的,都不会附带pdb文件。那么我们怎么能在程序崩溃的时候找到出错的具体位置呢?这个时候就该Dump文件出场了!Dump文件是进程的内存镜像,可以把程序运行时的状态完整的保存下来。
要想在程序崩溃的时候创建Dump文件,就需要用到DbgHelp.dll中Windows API的MiniDumpWriteDump()函数。该函数声明如下:
- __in HANDLE hProcess,
- __in DWORD ProcessId,
- __in HANDLE hFile,
- __in MINIDUMP_TYPE DumpType,
- __in PMINIDUMP_EXCEPTION_INFORMATION ExceptionParam,
- __in PMINIDUMP_USER_STREAM_INFORMATION UserStreamParam,
- __in PMINIDUMP_CALLBACK_INFORMATION CallbackParam
- );
具体的参数和返回值的解释可以查找MSDN,有很详细的说明。下面依然用上一篇文章中的例子代码来说明怎么在程序崩溃的时候创建Dump文件。
- // 处理Unhandled Exception的回调函数
- LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException)
- {
- // 这里弹出一个错误对话框并退出程序
- FatalAppExit(-1, _T("*** Unhandled Exception! ***"));
- return EXCEPTION_EXECUTE_HANDLER;
- }
- // 一个有函数调用的类
- class CrashTest
- {
- public:
- void Test()
- {
- Crash();
- }
- private:
- void Crash()
- {
- // 除零,人为的使程序崩溃
- int i = 13;
- int j = 0;
- int m = i / j;
- }
- };
- #include <Windows.h>
- #include <DbgHelp.h>
- // 添加对dbghelp.lib的编译依赖
- #pragma comment(lib, "dbghelp.lib")
- int _tmain(int argc, _TCHAR* argv[])
- {
- // 设置处理Unhandled Exception的回调函数
- SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);
- CrashTest test;
- test.Test();
- return 0;
- }
在上面的程序崩溃的时候,会调用函数ApplicationCrashHandler()。创建Dump文件的代码就需要添加到该函数中。下面就是一个创建Dump文件的函数。
- // 创建Dump文件
- void CreateDumpFile(LPCWSTR lpstrDumpFilePathName, EXCEPTION_POINTERS *pException)
- {
- // 创建Dump文件
- HANDLE hDumpFile = CreateFile(lpstrDumpFilePathName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
- // Dump信息
- MINIDUMP_EXCEPTION_INFORMATION dumpInfo;
- dumpInfo.ExceptionPointers = pException;
- dumpInfo.ThreadId = GetCurrentThreadId();
- dumpInfo.ClientPointers = TRUE;
- // 写入Dump文件内容
- MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
- CloseHandle(hDumpFile);
- }
在函数ApplicationCrashHandler()用类似下面的代码来调用上面的函数,就可以在程序崩溃的时候创建Dump文件。
下面简单说一下Dump文件的用法。将Dump文件拷贝到含有应用程序和对应的pdb文件的目录,在VS里面打开Dump文件(或者直接双击Dump文件),VS会自动创建一个Solution,直接调试运行,代码就会停到使程序崩溃的那一行上。就跟在VS里面调试代码一摸一样。(VS2008)
在VS2010里打开Dump文件,会显示一个Minidump File Summary,并且可以进行下面图中的操作。