C++/MFC全局未知异常捕获Dump出来并进行调试
全局捕获未知异常函数名:
WINBASEAPI
LPTOP_LEVEL_EXCEPTION_FILTER
WINAPI
SetUnhandledExceptionFilter(
__in_opt LPTOP_LEVEL_EXCEPTION_FILTER lpTopLevelExceptionFilter
);
该函数名声明在WinBase.h中 可以通过调用windows.h得到
这里给出一个demo作为演示
#include <windows.h>
//首先我们得先编写一个未知异常捕获的回调函数
LONG MyExceptionFilter(EXCEPTION_POINTERS *pExptInfo)
{
printf("Exception...");
return EXCEPTION_EXECUTE_HANDLER;
}
//注册异常回调函数
LPTOP_LEVEL_EXCEPTION_FILTER pPrevFilter = ::SetUnhandledExceptionFilter(MyExceptionFilter);
int main(int argc,char* argv[])
{
//我们在主函数中构造一个人为的异常出来
int j=0;
volatile int i=10/j; //我们都知道0不能当除数所以肯定抛异常出来 加volatile是因为防止变量未使用被编译器优化掉
return 1;
}
使用Visual Studio编译运行后如图 ,图中使用的是release模式生成的并使用控制台打开,因为如果用vs调试的会产生异常后会被IDE捕获到。
程序已经正确进入到异常处,由于在异常处,执行完后已经无法再回到异常现场,所以我们应该再异常处理完的时候直接退出程序。否则又不会抛一个异常出来(有时候不会抛出来 也没去深入研究是为什么)
接着我们对异常回调函数再进行改造,当异常发生的时候,程序退出之前,我们把异常现场dump出来方便后期进行调试定位。
直接给上完整代码
#include <iostream>
#include <windows.h>
#include <DbgHelp.h>
#pragma comment(lib,"DbgHelp.Lib") //MiniDumpWriteDump链接时用到
LONG WINAPI MyExceptionFilter(EXCEPTION_POINTERS *pExptInfo)
{
printf("Exception...");
// 程序崩溃时,将写入程序目录下的ExceptionDump.dmp文件
std::string path = "E:\\ExceptionDump.dmp";
size_t size = path.length();
wchar_t *buffer = new wchar_t[size + 1];
MultiByteToWideChar(CP_ACP, 0, path.c_str(), size, buffer, size * sizeof(wchar_t));
buffer[size] = 0; //确保以 '\0' 结尾
HANDLE hFile = ::CreateFile(buffer, GENERIC_WRITE,
FILE_SHARE_WRITE, NULL, CREATE_NEW,
FILE_ATTRIBUTE_NORMAL, NULL);
if (hFile != INVALID_HANDLE_VALUE)
{
MINIDUMP_EXCEPTION_INFORMATION exptInfo;
exptInfo.ThreadId = ::GetCurrentThreadId();
exptInfo.ExceptionPointers = pExptInfo;
//将内存堆栈dump到文件中
//MiniDumpWriteDump需引入dbghelp头文件
BOOL bOK = ::MiniDumpWriteDump(::GetCurrentProcess(),
::GetCurrentProcessId(),hFile, MiniDumpNormal,
&exptInfo, NULL, NULL);
}
return EXCEPTION_EXECUTE_HANDLER;
}
//注册异常回调函数
LPTOP_LEVEL_EXCEPTION_FILTER pPrevFilter = ::SetUnhandledExceptionFilter(MyExceptionFilter);
int main(int argc,char* argv[])
{
//我们在主函数中构造一个人为的异常出来
int j=0;
volatile int i=10/j; //我们都知道0不能当除数所以肯定抛异常出来 加volatile是因为防止变量未使用被编译器优化掉
return 1;
}
如果编译的时候出现找不到 DbgHelp.h 路径自行参考以下(编译器为32位):
C:\Program Files (x86)\Microsoft SDKs\Windows\v7.1A\Include
C:\Program Files (x86)\Microsoft SDKs\Windows\v7.1A\Lib
还有为了让pdb文件能够定位到源码中,需要再项目属性C/C++ –>常规中设置调试信息格式为Zi
然后回到控制台中运行,这时候显示跟第一次一样,但是在E盘中多了ExceptionDump.dmp,双击ExceptionDump.dmp,使用visual studio进行调试界面右边有个仅使用本机进行调试单机,如图: