1. 程式人生 > 其它 >QT+VS 呼叫基於Google Breakpad的跨平臺Qt崩潰異常捕獲呼叫方案

QT+VS 呼叫基於Google Breakpad的跨平臺Qt崩潰異常捕獲呼叫方案

方案一、基於Google Breakpad的跨平臺Qt崩潰異常捕獲呼叫方案

首先上部落格:Windows下Qt生成dump檔案並定位bug(基於qBreakpad)

這個地方使用的是一個叫qBreakPad的方案,這個之前在網上有過文件,但是文件太老了,不是很看得懂,這裡看到個說的比較明白的,故拿來簡單介紹一下:

1.生成靜態庫lib檔案

首先我們拿到這個qBreakpad工程檔案,先構建一下,生成一個我們想要的靜態庫檔案,可能會出現一些bug,在上文提到的部落格中有相關的一些解決方案。

這裡我們需要獲得兩個靜態庫檔案,一個是debug版本的,一個是release版本的,其實也可以只獲得需要的那個,但是我在這裡會上傳一個demo,裡面有兩個靜態庫檔案,具體存放方式如圖所示

1.

2.

3.include :

  1. 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();
    }