zoukankan      html  css  js  c++  java
  • 如何设置C++崩溃时生成Dump文件

    Dump 文件是进程的内存镜像 , 可以把程序的执行状态通过调试器保存到dump文件中 ; Dump 文件是用来给驱动程序编写人员调试驱动程序用的 , 这种文件必须用专用工具软件打开 , 比如使用 WinDbg , VisualStudio 打开 ;

    当我们的程序发布出去之后 , 在客户机上是无法跟踪自己代码的 BUG 的 , 所以 Dump 文件对于我们来说特别有用 ; 我们可以通过 .dmp 文件把出现 BUG 的情况再现 , 然后再现客户环境 (包括堆栈调用等情况) , 设置源码调试路径 , 可以找到出现 BUG 的语句 ;

    C++ 程序设置生成 Dump 文件的代码如下 :

    #include "stdafx.h"
    #include "Windows.h"
    #include "DbgHelp.h"
    
    int GenerateMiniDump(PEXCEPTION_POINTERS pExceptionPointers)
    {
        // 定义函数指针
        typedef BOOL(WINAPI * MiniDumpWriteDumpT)(
            HANDLE,
            DWORD,
            HANDLE,
            MINIDUMP_TYPE,
            PMINIDUMP_EXCEPTION_INFORMATION,
            PMINIDUMP_USER_STREAM_INFORMATION,
            PMINIDUMP_CALLBACK_INFORMATION
            );
        // 从 "DbgHelp.dll" 库中获取 "MiniDumpWriteDump" 函数
        MiniDumpWriteDumpT pfnMiniDumpWriteDump = NULL;
        HMODULE hDbgHelp = LoadLibrary(_T("DbgHelp.dll"));
        if (NULL == hDbgHelp)
        {
            return EXCEPTION_CONTINUE_EXECUTION;
        }
        pfnMiniDumpWriteDump = (MiniDumpWriteDumpT)GetProcAddress(hDbgHelp, "MiniDumpWriteDump");
    
        if (NULL == pfnMiniDumpWriteDump)
        {
            FreeLibrary(hDbgHelp);
            return EXCEPTION_CONTINUE_EXECUTION;
        }
        // 创建 dmp 文件件
        TCHAR szFileName[MAX_PATH] = {0};
        TCHAR* szVersion = _T("DumpDemo_v1.0");
        SYSTEMTIME stLocalTime;
        GetLocalTime(&stLocalTime);
        wsprintf(szFileName, L"%s-%04d%02d%02d-%02d%02d%02d.dmp",
            szVersion, stLocalTime.wYear, stLocalTime.wMonth, stLocalTime.wDay,
            stLocalTime.wHour, stLocalTime.wMinute, stLocalTime.wSecond);
        HANDLE hDumpFile = CreateFile(szFileName, GENERIC_READ | GENERIC_WRITE, 
            FILE_SHARE_WRITE | FILE_SHARE_READ, 0, CREATE_ALWAYS, 0, 0);
        if (INVALID_HANDLE_VALUE == hDumpFile)
        {
            FreeLibrary(hDbgHelp);
            return EXCEPTION_CONTINUE_EXECUTION;
        }
        // 写入 dmp 文件
        MINIDUMP_EXCEPTION_INFORMATION expParam;
        expParam.ThreadId = GetCurrentThreadId();
        expParam.ExceptionPointers = pExceptionPointers;
        expParam.ClientPointers = FALSE;
        pfnMiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), 
            hDumpFile, MiniDumpWithDataSegs, (pExceptionPointers ? &expParam : NULL), NULL, NULL);
        // 释放文件
        CloseHandle(hDumpFile);
        FreeLibrary(hDbgHelp);
        return EXCEPTION_EXECUTE_HANDLER;
    }
    
    LONG WINAPI ExceptionFilter(LPEXCEPTION_POINTERS lpExceptionInfo)
    {
        // 这里做一些异常的过滤或提示
        if (IsDebuggerPresent())
        {
            return EXCEPTION_CONTINUE_SEARCH;
        }
        return GenerateMiniDump(lpExceptionInfo);
    }
    
    int main()
    {
        // 加入崩溃dump文件功能
        SetUnhandledExceptionFilter(ExceptionFilter);
        // 使程序崩溃产生 Dump 文件
        int *p = NULL;
        *p=1;
    }

    然后编译项目 , 运行程序 , 就会发现在当前运行目录生成了 Dump 文件 , 类似 DumpDemo_v1.0-20170602-154446.dmp ;

    生成 Dump 文件后 , 我们可以用 VS2012 等工具来进行调试,传送门

  • 相关阅读:
    exp迁移测试库10.2.0.5
    DG_Check检测
    DG Switch over
    CPU查询
    记录数据库中,段大小的数据增长情况
    C++ 多态
    java反射
    git的基本概念
    实现MySQL的Replication
    网页只允许中国用户访问
  • 原文地址:https://www.cnblogs.com/zhanghu52030/p/9341799.html
Copyright © 2011-2022 走看看