[C++]Qt程式異常崩潰處理技巧(Win)

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

https://blog.csdn.net/baidu_33570760/article/details/52221863函數

 

這篇文章談的是 Qt4 程式在視窗系統下的異常崩潰處理技巧。因此需要在頭文件裏包括「#include <Windows.h>」。ui

首先,程式不免會有異常崩潰的時候。重要的是在崩潰時能及時把重要的數據保存好,將損失減小。spa

SetUnhandledExceptionFilter函數是Win32API的異常捕獲函數,在程式異常結束前。會調用該函數註冊的回調函數,這樣就能在進程終止前運行指定的代碼,達到好比保存數據的功能。.net

 1 LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException){//程式異常捕獲
 2     /*
 3       ***保存數據代碼***
 4     */
 5     //這裏彈出一個錯誤對話框並退出程序
 6     EXCEPTION_RECORD* record = pException->ExceptionRecord;
 7     QString errCode(QString::number(record->ExceptionCode,16)),errAdr(QString::number((uint)record->ExceptionAddress,16)),errMod;
 8     QMessageBox::critical(NULL,"程式崩潰","<FONT size=4><div><b>對於發生的錯誤,表示誠摯的歉意</b><br/></div>"+
 9         QString("<div>錯誤代碼:%1</div><div>錯誤地址:%2</div></FONT>").arg(errCode).arg(errAdr),
10         QMessageBox::Ok);
11     return EXCEPTION_EXECUTE_HANDLER;
12 }
13  
14 int main(int argc, char *argv[])
15 {
16     QApplication a(argc, argv);
17     QTextCodec::setCodecForTr(QTextCodec::codecForLocale());
18     QTextCodec::setCodecForCStrings(QTextCodec::codecForLocale());
19     QTextCodec::setCodecForLocale(QTextCodec::codecForLocale());
20     SetUnhandledExceptionFilter((LPTOP_LEVEL_EXCEPTION_FILTER)ApplicationCrashHandler);//註冊異常捕獲函數
21     MainWindow w;
22     w.showMaximized();
23     return a.exec();
24

保存數據僅是拯救措施,更重要的是找到錯誤的根源。若能在崩潰的同一時候,程式本身主動記錄下崩潰時的執行信息,將有助於修正工做。微軟提供了「DbgHelp」錯誤調試技術。調用相關功能就能夠保存程式崩潰時的信息,而後藉助WinDbg軟件就能分析出當時的執行情況。調試

調用「DbgHelp」的MiniDumpWriteDump函數保存以「.dmp」爲後綴的Dump文件,該文件能被WinDbg讀取並分析。code

你需要加入頭文件「#include <DbgHelp.h>」,在Pro文件里加入「LIBS += -lDbgHelp」。目的是連接DbgHelp庫。orm

 1 LONG ApplicationCrashHandler(EXCEPTION_POINTERS *pException){//程式異常捕獲
 2     /*
 3       ***保存數據代碼***
 4     */
 5     //建立 Dump 文件
 6     HANDLE hDumpFile = CreateFile(QTime::currentTime().toString("HH時mm分ss秒zzz.dmp").utf16(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
 7     if( hDumpFile != INVALID_HANDLE_VALUE){
 8         //Dump信息
 9         MINIDUMP_EXCEPTION_INFORMATION dumpInfo;
10         dumpInfo.ExceptionPointers = pException;
11         dumpInfo.ThreadId = GetCurrentThreadId();
12         dumpInfo.ClientPointers = TRUE;
13         //寫入Dump文件內容
14         MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hDumpFile, MiniDumpNormal, &dumpInfo, NULL, NULL);
15     }
16     //這裏彈出一個錯誤對話框並退出程序
17     EXCEPTION_RECORD* record = pException->ExceptionRecord;
18     QString errCode(QString::number(record->ExceptionCode,16)),errAdr(QString::number((uint)record->ExceptionAddress,16)),errMod;
19     QMessageBox::critical(NULL,"程式崩潰","<FONT size=4><div><b>對於發生的錯誤,表示誠摯的歉意</b><br/></div>"+
20         QString("<div>錯誤代碼:%1</div><div>錯誤地址:%2</div></FONT>").arg(errCode).arg(errAdr),
21         QMessageBox::Ok);
22     return EXCEPTION_EXECUTE_HANDLER;
23

當被錯誤困擾得焦頭爛額的時候。如果老天能直接告訴錯誤在哪一行代碼該有多好呀。其實WinDbg就能作到。htm

在項目的proproject文件裏增長:QMAKE_LFLAGS_RELEASE = /INCREMENTAL:NO /DEBUGblog

這句話的目的是Release版也將生成「.pdb」後綴的調試信息文件。在使用WinDbg導入Dump前。指定好源代碼與pdb文件的位置。就能夠在錯誤報告內看到罪魁禍首是哪一行代碼。

相關文章
相關標籤/搜索