QT+VS 呼叫基於Google Breakpad的跨平臺Qt崩潰異常捕獲呼叫方案
阿新 • • 發佈:2022-11-30
方案一、基於Google Breakpad的跨平臺Qt崩潰異常捕獲呼叫方案
首先上部落格:Windows下Qt生成dump檔案並定位bug(基於qBreakpad)
這個地方使用的是一個叫qBreakPad的方案,這個之前在網上有過文件,但是文件太老了,不是很看得懂,這裡看到個說的比較明白的,故拿來簡單介紹一下:
1.生成靜態庫lib檔案
首先我們拿到這個qBreakpad工程檔案,先構建一下,生成一個我們想要的靜態庫檔案,可能會出現一些bug,在上文提到的部落格中有相關的一些解決方案。
這裡我們需要獲得兩個靜態庫檔案,一個是debug版本的,一個是release版本的,其實也可以只獲得需要的那個,但是我在這裡會上傳一個demo,裡面有兩個靜態庫檔案,具體存放方式如圖所示
1.
2.
3.include :
- lib:
其中debug和release資料夾中只放了一個根據版本的qBreakpad.lib檔案
2.新增引用和庫
這個就不多說了,在那個vc++裡面設定一下就行了,記得這一步還需要在聯結器裡面新增lib檔案的引用,或者直接用pragma,也是可以的
3.寫程式碼
直接在main函式裡面,新增這麼一行
#include "QBreakpadHandler.h"
QBreakpadInstance.setDumpPath("crashes"); // 設定生成dump檔案路徑
這樣就可以在程式崩潰的時候生成dump檔案了
方案二,僅限於windows環境下的崩潰捕捉方案,呼叫更簡單
QDump.h
#pragma once //注意下面2個頭檔案引入的順序 #include <windows.h> #include <dbghelp.h> extern char* g_qsDumpName; class QtDumpGenerate { public: QtDumpGenerate(); void ApplicationCrashHandler(EXCEPTION_POINTERS *pException); }; // dump處理 // 使用方法: SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler); long ApplicationCrashHandler(EXCEPTION_POINTERS *pException);
QDump.cpp:
#include "QDump.h"
#include <QObject>
#include <QDir>
#include <QDateTime>
#include <QDebug>
#include <QCoreApplication>
#pragma comment(lib, "dbghelp.lib")
char* g_qsDumpName = "";
QtDumpGenerate::QtDumpGenerate()
{
}
void QtDumpGenerate::ApplicationCrashHandler(EXCEPTION_POINTERS *pException)
{
QDateTime current_date_time = QDateTime::currentDateTime();
QString current_date = current_date_time.toString("yyyyMMdd_hhmmss");
QString dmpName = QString::fromLocal8Bit(g_qsDumpName);
QString dmpFileName = QCoreApplication::applicationDirPath() + "/" + dmpName + current_date + ".dmp";
EXCEPTION_RECORD *record = pException->ExceptionRecord;
QString errCode(QString::number(record->ExceptionCode, 16));
QString errAddr(QString::number((uint)record->ExceptionAddress, 16));
QString errFlag(QString::number(record->ExceptionFlags, 16));
QString errPara(QString::number(record->NumberParameters, 16));
qDebug() << "errCode: " << errCode;
qDebug() << "errAddr: " << errAddr;
qDebug() << "errFlag: " << errFlag;
qDebug() << "errPara: " << errPara;
HANDLE hDumpFile = CreateFile((LPCWSTR)QString(dmpFileName).utf16(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (hDumpFile != INVALID_HANDLE_VALUE) {
MINIDUMP_EXCEPTION_INFORMATION dumpInfo;
dumpInfo.ExceptionPointers = pException;
dumpInfo.ThreadId = GetCurrentThreadId();
dumpInfo.ClientPointers = TRUE;
MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
CloseHandle(hDumpFile);
}
else {
qDebug() << "hDumpFile == null";
}
}
long ApplicationCrashHandler(EXCEPTION_POINTERS *pException)
{
QtDumpGenerate dmpGenerate;
dmpGenerate.ApplicationCrashHandler(pException);
return EXCEPTION_EXECUTE_HANDLER;
}
main.cpp:
#include "QCrushHandler.h"
#include <QtWidgets/QApplication>
#include "QDump.h"//方案二
#include "QBreakpadHandler.h"//方案一
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
//方案二
QString g_qsDumpName = "LgInteractSmartCompus";
SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);
//方案一
QBreakpadInstance.setDumpPath("crashes"); // 設定生成dump檔案路徑
QCrushHandler w;
w.show();
return a.exec();
}