赞
踩
参考 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; }
然后在main函数中执行
int main(int argc, char *argv[])
{
SetUnhandledExceptionFilter(errCallback);
QApplication app(argc, argv);
initMessageBox();
.....
}
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。