1. 程式人生 > >Qt程式異常崩潰處理技巧(Win)

Qt程式異常崩潰處理技巧(Win)

     文章轉載來自     http://www.cnblogs.com/lcchuguo/p/5177715.html     作者 lcchuguo

首先,程式難免會有異常崩潰的時候。重要的是在崩潰時能及時把重要的資料儲存好,將損失減少。

SetUnhandledExceptionFilter函式是Win32API的異常捕獲函式,在程式異常結束前。會呼叫該函式註冊的回撥函式,這樣就能在程序終止前執行指定的程式碼,達到比如儲存資料的功能。

LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException){//程式異常捕獲
    /*
      ***儲存資料程式碼***
    */
    //這裡彈出一個錯誤對話方塊並退出程式
    EXCEPTION_RECORD* record = pException->ExceptionRecord;
    QString errCode(QString::number(record->ExceptionCode,16)),errAdr(QString::number((uint)record->ExceptionAddress,16)),errMod;
    QMessageBox::critical(NULL,"程式崩潰","<FONT size=4><div><b>對於發生的錯誤,表示誠摯的歉意</b><br/></div>"+
        QString("<div>錯誤程式碼:%1</div><div>錯誤地址:%2</div></FONT>").arg(errCode).arg(errAdr),
        QMessageBox::Ok);
    return EXCEPTION_EXECUTE_HANDLER;
}

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    QTextCodec::setCodecForTr(QTextCodec::codecForLocale());
    QTextCodec::setCodecForCStrings(QTextCodec::codecForLocale());
    QTextCodec::setCodecForLocale(QTextCodec::codecForLocale());
    SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);//註冊異常捕獲函式
    MainWindow w;
    w.showMaximized();
    return a.exec();
}

儲存資料僅是拯救措施,更重要的是找到錯誤的根源。若能在崩潰的同一時候,程式自己主動記錄下崩潰時的執行資訊,將有助於修正工作。微軟提供了“DbgHelp”錯誤除錯技術。呼叫相關功能就可以儲存程式崩潰時的資訊,然後藉助WinDbg軟體就能分析出當時的執行狀況。

呼叫“DbgHelp”的MiniDumpWriteDump函式儲存以“.dmp”為字尾的Dump檔案,該檔案能被WinDbg讀取並分析。

你須要加入標頭檔案“#include <DbgHelp.h>”,在Pro檔案里加入“LIBS += -lDbgHelp”。目的是連結DbgHelp庫。

LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException){//程式異常捕獲
    /*
      ***儲存資料程式碼***
    */
    //建立 Dump 檔案
    HANDLE hDumpFile = CreateFile(QTime::currentTime().toString("HH時mm分ss秒zzz.dmp").utf16(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
    if( hDumpFile != INVALID_HANDLE_VALUE){
        //Dump資訊
        MINIDUMP_EXCEPTION_INFORMATION dumpInfo;
        dumpInfo.ExceptionPointers = pException;
        dumpInfo.ThreadId = GetCurrentThreadId();
        dumpInfo.ClientPointers = TRUE;
        //寫入Dump檔案內容
        MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
    }
    //這裡彈出一個錯誤對話方塊並退出程式
    EXCEPTION_RECORD* record = pException->ExceptionRecord;
    QString errCode(QString::number(record->ExceptionCode,16)),errAdr(QString::number((uint)record->ExceptionAddress,16)),errMod;
    QMessageBox::critical(NULL,"程式崩潰","<FONT size=4><div><b>對於發生的錯誤,表示誠摯的歉意</b><br/></div>"+
        QString("<div>錯誤程式碼:%1</div><div>錯誤地址:%2</div></FONT>").arg(errCode).arg(errAdr),
        QMessageBox::Ok);
    return EXCEPTION_EXECUTE_HANDLER;
}

當被錯誤困擾得焦頭爛額的時候。若是老天能直接告訴錯誤在哪一行程式碼該有多好呀。其實WinDbg就能做到。

在專案的proproject檔案裡增加:QMAKE_LFLAGS_RELEASE = /INCREMENTAL:NO /DEBUG

這句話的目的是Release版也將生成“.pdb”字尾的除錯資訊檔案。在使用WinDbg匯入Dump前。指定好原始碼與pdb檔案的位置。就可以在錯誤報告內看到罪魁禍首是哪一行程式碼。