[转]让程序在崩溃时体面的退出之SEH+Dump文件

原文地址:http://blog.csdn.net/starlee/article/details/6649605

在我上篇文章《让程序在崩溃时体面的退出之SEH》中讲解了SEH中try/except可以捕捉异常,避免程序的崩溃,并且可以在处理完异常之后,还能决定进该进程如何执行。对于应用程序的使用者来说,并不知道异常的发生。但是对于软件的开发者来说,虽然避免了程序的崩溃,可是这样可以让程序崩溃的缺陷存在于代码中,就像一个定时炸弹,不知道什么时候会爆炸。要想修复这样的缺陷,首先要找到导致程序崩溃的那行代码。而我在我的那篇《让程序在崩溃时体面的退出之Dump文件》里面介绍了如何用Dump文件来定位使程序崩溃的代码。这里依然可以用同样的方法。下面就是创建Dump文件的函数。

  1. // 创建Dump文件
  2. //
  3. void CreateDumpFile(LPCWSTR lpstrDumpFilePathName, EXCEPTION_POINTERS *pException)
  4. {
  5. // 创建Dump文件
  6. //
  7. HANDLE hDumpFile = CreateFile(lpstrDumpFilePathName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
  8. // Dump信息
  9. //
  10. MINIDUMP_EXCEPTION_INFORMATION dumpInfo;
  11. dumpInfo.ExceptionPointers = pException;
  12. dumpInfo.ThreadId = GetCurrentThreadId();
  13. dumpInfo.ClientPointers = TRUE;
  14. // 写入Dump文件内容
  15. //
  16. MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
  17. CloseHandle(hDumpFile);
  18. }

从上面的代码中可以看出,要想创建Dump文件,必须得到一个指向EXCEPTION_POINTERS结构的指针。怎么在try/except块中得到这个指针呢?这个时候就需要用到Windows API中的GetExceptionInformation()。这个函数的返回值就是一个指向EXCEPTION_POINTERS结构的指针。下面是具体的代码。

  1. // 作为except块中表达式的函数
  2. //
  3. LONG CrashHandler(EXCEPTION_POINTERS *pException)
  4. {
  5. // 在这里添加处理程序崩溃情况的代码
  6. //
  7. // 这里以弹出一个对话框为例子
  8. //
  9. MessageBox(NULL, _T("Message from Catch handler"), _T("Test"), MB_OK);
  10. // 创建Dump文件
  11. //
  12. CreateDumpFile(_T("C:\\Test.dmp"), pException);
  13. return EXCEPTION_EXECUTE_HANDLER;
  14. }
  15. int _tmain(int argc, _TCHAR* argv[])
  16. {
  17. __try
  18. {
  19. MessageBox(NULL, _T("Message from '__try' section"), _T("Test"), MB_OK);
  20. // 除零,人为的使程序崩溃
  21. //
  22. int i = 13;
  23. int j = 0;
  24. int m = i / j;
  25. }
  26. // 捕捉到让程序崩溃的异常时创建Dump文件
  27. //
  28. __except(CrashHandler(GetExceptionInformation()))
  29. {
  30. // 这里以弹出一个对话框为例子
  31. //
  32. MessageBox(NULL, _T("Message from '__except' section"), _T("Test"), MB_OK);
  33. }
  34. MessageBox(NULL, _T("Funcation completed"), _T("Test"), MB_OK);
  35. return 0;
  36. }

编译上面的代码并运行,会依次弹出下面这些对话框,并在C盘创建一个Dump文件Test.dmp。

[转]让程序在崩溃时体面的退出之SEH+Dump文件[转]让程序在崩溃时体面的退出之SEH+Dump文件[转]让程序在崩溃时体面的退出之SEH+Dump文件[转]让程序在崩溃时体面的退出之SEH+Dump文件

有了Dump文件,就可以轻松定位使程序崩溃的那行代码,具体方法可参考我的《让程序在崩溃时体面的退出之Dump文件》。

上一篇:DevOps实践


下一篇:java并发包小结(二)