当前位置:   article > 正文

Qt(windows下)捕获异常信息并自动重启_qt获取重启消息

qt获取重启消息

参考 https://blog.csdn.net/x85371169/article/details/79267592

目前在弄一个工业上用的软件,需要实现无人值守功能。软件经过两三个星期的debug,已经将绝大部分导致软件死机的bug修复,但是由于能力有限,软件还是会在某些情况下(我也不知道啥情况。。)死机,因此只能弄个错误捕获,然后自动重启了。

windows下的异常捕获过程可以参考文章开头那个博文。但是那篇文章中的QMessageBox,会有无法点击的问题(假如异常是在子线程触发的)。
下面的代码是对该情况的修复,顺便增加了个重启功能。

#include <DbgHelp.h>
QMessageBox *msgbPtr;//在子线程内创建的messageBox会有问题,所以只能在主线程创建并传递过过来
void initMessageBox()
{
    msgbPtr = new QMessageBox();
    msgbPtr->setWindowFlag(Qt::WindowStaysOnTopHint);
    msgbPtr->setIcon(QMessageBox::Critical);
    msgbPtr->setWindowTitle("软件崩溃");
    msgbPtr->setText("未知原因崩溃,程序将在5s后重启");

    QPushButton *rebootButton;
    QPushButton *rejectButton ;

    rebootButton = msgbPtr->addButton("重启", QMessageBox::AcceptRole);
    rejectButton = msgbPtr->addButton("退出", QMessageBox::RejectRole);

    QObject::connect(msgbPtr, &QMessageBox::buttonClicked, [=](QAbstractButton *button){
        if(msgbPtr->clickedButton() == (QAbstractButton*)rebootButton)
        {
            qDebug() << "start application:" << QProcess::startDetached(qApp->applicationFilePath(), QStringList());//重启
//            qApp->quit();
        }

        msgbPtr->close();
    });
}

//异常捕获回调函数
long __stdcall   errCallback(_EXCEPTION_POINTERS*  pException)
{
/******这里最好还是加个防止反复进入该回调函数的机制。防止在倒计时的过程中,程序那边又再次触发异常。*********/
    // 信号量的意义,把操作共享内存的代码锁住。因为有可能同时启动, 防止并发
    QSystemSemaphore sema("my exception", 1, QSystemSemaphore::Open);
    sema.acquire();
 
    QDir dir;
    dir.mkdir("../dumps");
    dir.cd("../dumps");

    /*
      ***保存数据代码***
    */

    QString fileName = dir.path() + "/" +
                       QDateTime::currentDateTime().toString("yyyyMMdd_HHmmss.zzz") + ".dmp";

    LPCWSTR pFileName = (LPCWSTR)fileName.unicode();

    //创建 Dump 文件
    HANDLE hDumpFile = CreateFile(pFileName,
                                  GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
    qDebug() << "create dumpFile:" << hDumpFile << INVALID_HANDLE_VALUE;
    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;


    qDebug() << "main thread:" << qApp->thread() << QThread::currentThread();

    int lastTime = 5;
    QTimer *timer = new QTimer();
    timer->setInterval(1000);
    QObject::connect(timer, &QTimer::timeout, [&lastTime, &errCode](){
        lastTime--;

        if(lastTime < 0)
        {
            //            qApp->quit();
            qDebug() << "start application:" << QProcess::startDetached(qApp->applicationFilePath(), QStringList());//重启
            msgbPtr->close();
            return;
        }

        QString info = QString("未知原因崩溃,程序将在%1s后重启\r\n%2").arg(lastTime).arg(errCode);
        msgbPtr->setText(info);
    });
    timer->start();

    msgbPtr->exec();


    return EXCEPTION_EXECUTE_HANDLER;
}

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94

然后在main函数中执行

int main(int argc, char *argv[])
{
    SetUnhandledExceptionFilter(errCallback);
    QApplication app(argc, argv);
    initMessageBox();

    .....
    
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小惠珠哦/article/detail/982179
推荐阅读
相关标签
  

闽ICP备14008679号