• 欢迎浏览“String me = Creater\忠实的资深Linux玩家;”,请文明浏览,理性发言,有侵犯你的权益请邮件我(creater@vip.qq.com).
  • 把任何的失败都当作一次尝试,不要自卑;把所有的成功都想成是一种幸运,不要自傲。
  •    4年前 (2015-01-04)  QT |   抢沙发  23 
    文章评分 0 次,平均分 0.0

    以前记录过C#开发应用程序时程序异常崩溃时的调试记录程序,链接为:C#开发的应用程序:自动退出原因查找

    在Qt中首先需要在头文件中包含“#include
    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读取并分析。

    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就能做到。
    在项目的pro工程文件中加入:QMAKE_LFLAGS_RELEASE = /INCREMENTAL:NO /DEBUG
    这句话的目的是Release版也将生成“.pdb”后缀的调试信息文件。在使用WinDbg导入Dump前,指定好源码与pdb文件的位置,即可在错误报告内看到罪魁祸首是哪一行代码。

     

    除特别注明外,本站所有文章均为String me = "Creater\忠实的资深Linux玩家";原创,转载请注明出处来自http://unix8.net/home.php/3912.html

    关于

    发表评论

    暂无评论

    切换注册

    登录

    忘记密码 ?

    切换登录

    注册

    扫一扫二维码分享