本文只是记录下 MSVC+Qt 生成 dump 的代码。
dump 文件能够保存程序内部的内存、堆栈、句柄、线程等程序运行相关的信息,当程序异常无法在调试环境里解决时,dump 文件是分析问题的重要手段。
相关 win API 文档可以在 MSDN 查看:https://docs.microsoft.com/zh-cn/search/?terms=MiniDumpWriteDump
DUMPTYPE 参数我只使用了 MiniDumpNormal,可以根据文档加上自己需要的。
dump 文件一般配合 pdb 文件使用,通过 pdb 可以找到堆栈地址对应的函数、行号等。
如果使用的 QtCreator,可以打开如下设置:
如果使用 VS ,可以打开如下两个设置:
生成 dump 完整代码:
#include "mainwindow.h"
#include "CrashDump.h"
#include <QApplication>
int main(int argc, char *argv[])
{
CrashDump::init("Test");
QApplication a(argc, argv);
MainWindow w;
w.show();
return a.exec();
}
#pragma once
#include <QObject>
#include <QCalendar>
#include <QDateTime>
#include <QFileInfo>
#include <QDir>
#include <Windows.h>
#include <DbgHelp.h>
#pragma comment(lib, "DbgHelp")
#include <iostream>
static QString __module_name;
class CrashDump
{
public:
//初始化注册
static void init(const QString &module)
{
__module_name=module;
//异常捕获
::SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)exceptionFilter);
}
//获取exe路径
static QString getAppPath()
{
DWORD v;
QVarLengthArray<char, 2048> buffer;
size_t size = 0;
do {
size += 2048;
buffer.resize((int)size);
v = ::GetModuleFileNameA(NULL, buffer.data(), DWORD(buffer.size()));
} while (v >= size);
return QString::fromLocal8Bit(buffer.data(), v);
}
//生成dump路径,可以自定义路径以及命名格式
static QString getDumpPath()
{
QFileInfo app_info(getAppPath());
QString dump_path=QString("%1/dump/%2 %3.dmp")
.arg(app_info.absolutePath())
.arg(__module_name)
.arg(QDateTime::currentDateTime().toString("yyyy_MM_dd hh.mm.ss"));
QFileInfo dump_info(dump_path);
if(!dump_info.dir().exists()&&!dump_info.dir().mkpath(dump_info.absolutePath()))
dump_path=QString("%1.dump").arg(__module_name);
return dump_path;
}
//异常处理
static LONG exceptionFilter(LPEXCEPTION_POINTERS lpExceptionInfo)
{
QString dump_path = getDumpPath();
std::cerr<<"crash dump:"<<dump_path.toStdString()<<std::endl;
//创建文件
HANDLE file_handle = ::CreateFileA(dump_path.toLocal8Bit().data(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (file_handle != INVALID_HANDLE_VALUE) {
DWORD dump_type = MiniDumpNormal; //可以加上需要的其他枚举值
MINIDUMP_EXCEPTION_INFORMATION dump_info;
dump_info.ClientPointers = TRUE;
dump_info.ExceptionPointers = lpExceptionInfo;
dump_info.ThreadId = ::GetCurrentThreadId();
//写入dump
::MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), file_handle, (MINIDUMP_TYPE)dump_type, &dump_info, NULL, NULL);
}
::CloseHandle(file_handle);
//这里可以把程序拉起来
return EXCEPTION_CONTINUE_SEARCH; //or EXCEPTION_EXECUTE_HANDLER
}
};
程序遇到异常生成 dump 文件后,可以使用 VS 打开:
这里符号路径我只设置了 exe 对应 pdb 的路径:
最后点仅限本机调试就可以看到异常现场了:
(2021-3-26)要注意的是,SetUnhandledExceptionFilter 并不能捕获所有的异常,可以搜索其他的相关函数:
//terminate()
set_terminate(terminateHandler);
//printf(NULL)
_set_invalid_parameter_handler(invalidParameterHandler);
//调用纯虚函数
_set_purecall_handler(purecallHandler);
//new异常
_set_new_handler(newHandler);
其他
-
国内开源:https://gitee.com/feiyangqingyun
-
国际开源:https://github.com/feiyangqingyun
-
项目大全:https://qtchina.blog.csdn.net/article/details/97565652