文章轉載來自 http://www.cnblogs.com/lcchuguo/p/5177715.html 作者 lcchuguo
https://blog.csdn.net/baidu_33570760/article/details/52221863
這篇文章談的是 Qt4 程式在視窗系統下的異常崩潰處理技巧。所以須要在頭文件里包括“#include <Windows.h>”。
首先,程式難免會有異常崩潰的時候。重要的是在崩潰時能及時把重要的數據保存好,將損失減少。
SetUnhandledExceptionFilter函數是Win32API的異常捕獲函數,在程式異常結束前。會調用該函數注冊的回調函數,這樣就能在進程終止前運行指定的代碼,達到比如保存數據的功能。
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讀取並分析。
你須要加入頭文件“#include <DbgHelp.h>”,在Pro文件里加入“LIBS += -lDbgHelp”。目的是鏈接DbgHelp庫。
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就能做到。
在項目的proproject文件里增加:QMAKE_LFLAGS_RELEASE = /INCREMENTAL:NO /DEBUG
這句話的目的是Release版也將生成“.pdb”后綴的調試信息文件。在使用WinDbg導入Dump前。指定好源代碼與pdb文件的位置。就可以在錯誤報告內看到罪魁禍首是哪一行代碼。