zoukankan      html  css  js  c++  java
  • [C++]Qt程式异常崩溃处理技巧(Win)

    文章转载来自     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文件的位置。就可以在错误报告内看到罪魁祸首是哪一行代码。

  • 相关阅读:
    shell脚本修改Linux系统中所有IP样例
    关闭并卸载数据库脚本
    查询编译不通过的存储过程并重新编译
    SQL函数造数据样例(一)
    类型转换和多态
    Java学习笔记(三)
    Java学习笔记二()
    Java学习笔记(一)
    1.2.零宽断言
    1.3.匹配小括号的字符(可能有小括号在一行的,也有多行的)
  • 原文地址:https://www.cnblogs.com/FKdelphi/p/10731775.html
Copyright © 2011-2022 走看看