Qt下生成pdb文件,并在exe崩溃时生成dmp文件,且由dmp查询崩溃原因

2024-02-08 15:40

本文主要是介绍Qt下生成pdb文件,并在exe崩溃时生成dmp文件,且由dmp查询崩溃原因,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、编译时生成pdb文件

选择profile编译即可,在此模式下,既可享受release的运行速度,又可以享受debug的符号查找。
发布的程序可以正常运行在生产环境,尤其是适合在一些实时性要求高的项目中。
在这里插入图片描述

二、在exe崩溃时生成dmp文件

参考:
https://gongjianbo1992.blog.csdn.net/article/details/113791423
https://blog.csdn.net/a844651990/article/details/85225273
https://www.lmlphp.com/user/62390/article/item/902039/

声明函数

DumpHelper.h

//#ifndef DUMPHELPER_H
//#define DUMPHELPER_H#pragma once#include <QSystemSemaphore>
#include <QDir>
#include <QDateTime>
#include <QDebug>#include <tchar.h>
#include <Windows.h>
#include <DbgHelp.h>#pragma comment(lib, "user32.lib")
#pragma comment(lib, "DbgHelp.Lib")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[] = L"DumpFile";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);
}long __stdcall errCallback(_EXCEPTION_POINTERS*  pException)
{//     //用于崩溃重启
//    // 信号量的意义,把操作共享内存的代码锁住。因为有可能同时启动, 防止并发
//    QSystemSemaphore sema("DyError", 1, QSystemSemaphore::Open);
//    sema.acquire();QDir dir;dir.mkdir("./dumps");dir.cd("./dumps");/****保存数据代码****/QString fileName = dir.path() + "/" +QDateTime::currentDateTime().toString("yyyyMMdd_HHmmss.zzz") + ".dmp";LPCWSTR pFileName = (LPCWSTR)fileName.unicode();//创建 Dump 文件HANDLE hDumpFile = CreateFile(pFileName,GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);qDebug() << "create dumpFile:" << hDumpFile << INVALID_HANDLE_VALUE;if(hDumpFile != INVALID_HANDLE_VALUE){//Dump信息MINIDUMP_EXCEPTION_INFORMATION dumpInfo;dumpInfo.ExceptionPointers = pException;dumpInfo.ThreadId = GetCurrentThreadId();dumpInfo.ClientPointers = TRUE;// ::全局作用域符号// 写入Dump文件内容// DumpType这里仅仅保存普通信息。假如需要保存变量值,可以加上【MiniDumpWithFullMemory】// 参考https://learn.microsoft.com/en-us/windows/win32/api/minidumpapiset/ne-minidumpapiset-minidump_type::MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
//        ::MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal | MiniDumpWithFullMemory, &dumpInfo, NULL, NULL);}//    delete  unimem;//    qDebug() << "start application:" << QProcess::startDetached(qApp->applicationFilePath(), QStringList());//重启//    qApp->quit();qApp->exit(-1);return EXCEPTION_EXECUTE_HANDLER;
}//#endif // DUMPHELPER_H

调用函数

在main函数中注册

main.cpp

#include "mainwindow.h"#include <QApplication>#include "DumpHelper.h"int main(int argc, char *argv[])
{
//    SetUnhandledExceptionFilter(ExceptionFilter); // dmp文件比较大SetUnhandledExceptionFilter(errCallback); // dmp文件比较小QApplication a(argc, argv);MainWindow w;w.show();return a.exec();
}

三、经由dmp文件查询崩溃原因

在我们写好程序,编译好,且发布了给客户时,一定要注意保存几个东西:此时的源码、此时的exe+pdb文件。
在给到客户的程序发生了崩溃,产生了dmp文件后,我们需要把我们当时发布的exe、pdb、客户发给我们的dmp文件放在一起,然后用vs打开,进行调试操作。

基本前提

能够查询溯源的前提是对软件发布时的东西做好了备份:源码、pdb、exe
下面以实例说明。
目前,有一个 versionTest 项目,经过编译后,发布了v1版本。内容如下:
在这里插入图片描述

并且,在发布时,我也备份了对应的源码、pdb文件
在这里插入图片描述

假设此时,用户发现软件崩溃了,在对应目录生成了dmp文件,他将文件发给了我。

1.将对应的dmp文件、exe文件、pdb文件拷贝你发布的一份程序目录下

在这里插入图片描述

2.用vs打开dmp文件,并执行调试

在这里插入图片描述在这里插入图片描述
这个最好是等他下载完(这次下载完了会缓存起来,下次就可以直接打开了),否则有些跨线程的bug无法被精确定位。
关于qt自己的pdb文件,可以参考这篇文章。同时,最好在安装Qt时就选择安装qt的源码,因为Qt的pdb文件会指向qt的源文件。
此时它可能会定位到你最新的代码那里去的。

3.自己手动定位一下真正源码位置

在这里插入图片描述在这里插入图片描述
选择你之前备份的文件夹
在这里插入图片描述
自然就对上了。

qBreakPad

有个开源库可以做到跨平台生成dump,我还没实际测试过。有兴趣的可以试试。
https://blog.csdn.net/CLinuxF/article/details/122944959

用dmp文件查找错误时卡住

有时后可能是因为所使用的第三方库的问题,导致一直卡在第三方库。此时就需要自己把第三方库的名字改一下,让在索引时跳过第三方库。
比如我在使用思谋的sdk后,假如想根据程序崩溃时生成的dmp文件找到出错的原因,那就在前面的设置的基础上,将exe目录下的 vimo_inference.dll 文件 改成 vimo_inference-.dll
在这里插入图片描述

这篇关于Qt下生成pdb文件,并在exe崩溃时生成dmp文件,且由dmp查询崩溃原因的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/691481

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

数据库oracle用户密码过期查询及解决方案

《数据库oracle用户密码过期查询及解决方案》:本文主要介绍如何处理ORACLE数据库用户密码过期和修改密码期限的问题,包括创建用户、赋予权限、修改密码、解锁用户和设置密码期限,文中通过代码介绍... 目录前言一、创建用户、赋予权限、修改密码、解锁用户和设置期限二、查询用户密码期限和过期后的修改1.查询用

MybatisGenerator文件生成不出对应文件的问题

《MybatisGenerator文件生成不出对应文件的问题》本文介绍了使用MybatisGenerator生成文件时遇到的问题及解决方法,主要步骤包括检查目标表是否存在、是否能连接到数据库、配置生成... 目录MyBATisGenerator 文件生成不出对应文件先在项目结构里引入“targetProje

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

Python使用qrcode库实现生成二维码的操作指南

《Python使用qrcode库实现生成二维码的操作指南》二维码是一种广泛使用的二维条码,因其高效的数据存储能力和易于扫描的特点,广泛应用于支付、身份验证、营销推广等领域,Pythonqrcode库是... 目录一、安装 python qrcode 库二、基本使用方法1. 生成简单二维码2. 生成带 Log

基于Qt Qml实现时间轴组件

《基于QtQml实现时间轴组件》时间轴组件是现代用户界面中常见的元素,用于按时间顺序展示事件,本文主要为大家详细介绍了如何使用Qml实现一个简单的时间轴组件,需要的可以参考下... 目录写在前面效果图组件概述实现细节1. 组件结构2. 属性定义3. 数据模型4. 事件项的添加和排序5. 事件项的渲染如何使用

MySQL不使用子查询的原因及优化案例

《MySQL不使用子查询的原因及优化案例》对于mysql,不推荐使用子查询,效率太差,执行子查询时,MYSQL需要创建临时表,查询完毕后再删除这些临时表,所以,子查询的速度会受到一定的影响,本文给大家... 目录不推荐使用子查询和JOIN的原因解决方案优化案例案例1:查询所有有库存的商品信息案例2:使用EX

Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南

《Python使用Pandas库将Excel数据叠加生成新DataFrame的操作指南》在日常数据处理工作中,我们经常需要将不同Excel文档中的数据整合到一个新的DataFrame中,以便进行进一步... 目录一、准备工作二、读取Excel文件三、数据叠加四、处理重复数据(可选)五、保存新DataFram

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P

SpringBoot基于MyBatis-Plus实现Lambda Query查询的示例代码

《SpringBoot基于MyBatis-Plus实现LambdaQuery查询的示例代码》MyBatis-Plus是MyBatis的增强工具,简化了数据库操作,并提高了开发效率,它提供了多种查询方... 目录引言基础环境配置依赖配置(Maven)application.yml 配置表结构设计demo_st