C++/MFC全局未知异常捕获并进行调试

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进行调试界面右边有个仅使用本机进行调试单机,如图:
这里写图片描述
这里写图片描述

注意:要显示源码必须有PDB文件,还有尽量不要优化代码 不然定位出来可能位置有问题,还有一点再异常中别搞太多事了,不然再出异常就没地方抛了XXXX对话框就出来了。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值