Qt:处理后台线程处理程序抛出的异常

Scu*_*der 5 c++ qt multithreading exception qt4

我知道您需要重新实现 QApplication::notify() 方法才能正确捕获从主线程的事件处理程序抛出的异常。但是其他线程呢?假设我有一个带有槽的对象,并且该对象位于 QThread 中(具有默认的 run() 方法,该方法仅调用 exec()),即该对象的线程关联性是后台 QThread。那么,我应该在哪里捕获从该对象的槽抛出的异常呢?

IOW,如何重新实现后台线程的notify()方法?

san*_*ood 3

当您使用覆盖的通知方法创建自定义应用程序时;一旦您创建的 QThread 启动了自己的事件循环,它也会使用此重写方法(作为主线程)

这意味着在实践中,如果将任何插槽连接到 QThread::started 信号;那么这个槽在线程的事件循环之外执行,因此不在重写的通知方法中。

以下是有助于理解发生的情况的代码示例:

#include "mainwindow.h"
#include <QApplication>

#include <QDebug>
#include <QThread>
#include <QTimer>
#include <exception>

class ThrowingObject : public QObject
{
public:
    void doThrowInNotify()
    {
        qDebug() << "I am execution on thread id" << QThread::currentThreadId() << " and I am throwing";
        throw std::exception("KBOOOM");
    }

    void scheduleThrow()
    {
        QTimer* singleShot = new QTimer(this);
        singleShot->setSingleShot(true);

        connect(singleShot, &QTimer::timeout, this, &ThrowingObject::doThrow);

        singleShot->start();

        qDebug() << "I am execution on thread id" << QThread::currentThreadId() << " and I will throw in run";
    }

    void doThrow()
    {
        qDebug() << "I am execution on thread id" << QThread::currentThreadId() << " and I am throwing right now";

        //This exception is not catched, and definitly crash the process
        throw std::exception("KBOOOM");
    }

    void doThrowOutsideNotify()
    {
        //wait 5s for demo purpose, this slot is called by Object2, after Object1 throw in thread1 event loop
        QThread::sleep(5);
        qDebug() << "I am execution on thread id" << QThread::currentThreadId() << " and I am throwing right now";

        //This exception is not catched, and definitly crash the process
        throw std::exception("FATAL KBOOOM");
    }

};

class ApplicationWithExceptionCatchedInNotify : public QApplication
{
public:
    ApplicationWithExceptionCatchedInNotify(int argc, char *argv[]) :
        QApplication(argc,argv)
    {}

    bool notify(QObject* receiver, QEvent *e) override
    {
        try {
            return QApplication::notify(receiver, e);
        }
        catch(std::runtime_error e)
        {
            qDebug() << "std::runtime_error in thread : " << QThread::currentThreadId();
            qDebug() << e.what();
        }
        catch(std::exception e)
        {
            qDebug() << "std::exception in thread : " << QThread::currentThreadId();
            qDebug() << e.what();
        }
        catch(...)
        {
            qDebug() << "exception thread : " << QThread::currentThreadId();
        }

        qDebug() << "catch in notify ";
        return false;
    }

};



int main(int argc, char *argv[])
{
ApplicationWithExceptionCatchedInNotify app(argc, argv);

qDebug() << "Main QThread id" << QThread::currentThreadId();

//Object o1 will throw in its event loop (in notify)
QThread thread1;
ThrowingObject o1;
o1.moveToThread(&thread1);

QObject::connect(&thread1, &QThread::started, &o1, &ThrowingObject::scheduleThrow);

thread1.start();

//Object o2 will throw before the event loop is installed
QThread thread2;
ThrowingObject o2;
o2.moveToThread(&thread2);
//Connect to started signal.
QObject::connect(&thread2, &QThread::started, &o2, &ThrowingObject::doThrowOutsideNotify);
thread2.start();

app.exec();
}
Run Code Online (Sandbox Code Playgroud)

在 Windows 上运行此代码示例,Qt 5.9 和 Qt Creator 给出的示例如下:

输出 :

Main QThread id 0x11e4
I am execution on thread id 0x180c  and I will throw in run
I am execution on thread id 0x180c  and I am throwing right now
std::exception in thread :  0x180c
KBOOOM
catch in notify 
I am execution on thread id 0x27b8  and I am throwing right now
Run Code Online (Sandbox Code Playgroud)

一个 Microsoft Visual Studio Runtime Library 弹出来:

Microsoft Visual C++ Runtime Library

Debug Error!

Program: ...ad-Desktop_Qt_5_9_2_MSVC2017_64bit-Debug\debug\DemoThread.exe

abort() has been called

(Press Retry to debug the application)
Run Code Online (Sandbox Code Playgroud)

如果你设置断点;一旦可以意识到:

对象 o1 抛出 QThread::exec 方法,沿着调用堆栈我们可以看到我们处于 ApplicationWithExceptionCatchedInNotify::Notify 中

1 ThrowingObject::doThrow main.cpp 35 0x7ff66615352b 2 QtPrivate::FunctorCall,QtPrivate::List<>,void,void (__cdecl ThrowingObject:: *)(void) __ptr64>::call qobjectdefs_impl.h 136 0x7ff66615358c 3 QtPrivate:: FunctionPointer::call,void> qobjectdefs_impl.h 170 0x7ff666152ce7 4​​ QtPrivate::QSlotObject,void>::impl qobject_impl.h 121 0x7ff66615363e 5 QtPrivate::QSlotObjectBase::call qobject_impl.h 101 0x54a82428
6 QMetaObject::activate qobject.cpp 3754 0x54a70ee0
7 QMetaObject::激活 qobject.cpp 3629 0x54a707a8
8 QTimer::超时 moc_qtimer.cpp 202 0x54a8f739
9 QTimer::timerEvent qtimer.cpp 257 0x54a8f79a
10 QObject::事件 qobject.cpp 1228 0 x54a72b73
11 QApplicationPrivate::notify_helper qapplication.cpp 3722 0x53aeb8ee
12 QApplication::notify qapplication.cpp 3094 0x53ae6323
13 ApplicationWithExceptionCatchedInNotify::notify main.cpp 60 0x7ff666158730 14 QCoreApplication::notifyInternal2 qcoreapplication.cpp 1018 0x54a1b0c6
15 QCoreApplication::sendEvent qcoreapplication .h 233 0x54a26062
16 QEventDispatcherWin32::事件 qeventdispatcher_win.cpp 1041 0x54ad8cab
17 QApplicationPrivate::notify_helper qapplication.cpp 3722 0x53aeb8ee
18 QApplication::notify qapplication.cpp 3094 0x53ae6323
19 ApplicationWithExceptionCatchedInNotify::notify main.cpp 60 0x7ff666158730 20 QCoreApplication::notifyInternal2 qcoreapplication.cpp 1018 0x54a1b0c6
21 QCoreApplication::sendEvent qcoreapplication.h 233 0x 54a26062
22 QCoreApplicationPrivate::sendPostedEvents qcoreapplication.cpp 1678 0x54a1c982
23 QEventDispatcherWin32::sendPostedEvents qeventdispatcher_win.cpp 1064 0x54ad8e6a
24 qt_internal_proc qeventdispatcher_win.cpp 237 0x54ad6b47
25 CallWindowProcW USER32 0x7ffba1571c24 26 DispatchMessageW USER32 0x7ffba157 156c 27 QEventDispatcherWin32::processEvents qeventdispatcher_win.cpp 628 0x54ad755b
28 QEventLoop::processEvents qeventloop.cpp 135 0x54a15498
29 QEventLoop::exec qeventloop.cpp 212 0x54a156de
30 QThread::exec qthread.cpp 515 0x5465028f
31 QThread::run qthread.cpp 583 0x546501c3
32 QThreadPrivate::start qthread_win.cpp 380 0x5465caed
33 BaseThreadInitThunk KERNEL32 0x7ffb9f5b8364 34 RtlUserThreadStart ntdll 0x7ffba1bb7091

对象 o2 抛出 QThread::start 方法;在 thread2 事件循环之外

1 throwingObject :: dothrowoutsideNotify main.cpp 45 0x7ff666152ba6 2 qtprivate :: FunctorCall,qtprivate,qtprivate :: List <> list <> void,void(__cdecl throwobject :::: *) : FunctionPointer::call,void> qobjectdefs_impl.h 170 0x7ff666152ce7 4​​ QtPrivate::QSlotObject,void>::impl qobject_impl.h 121 0x7ff66615363e 5 QtPrivate::QSlotObjectBase::call qobject_impl.h 101 0x54a82428
6 QMetaObject::activate qobject.cpp 3754 0x54a70ee0
7 QMetaObject::激活 qobject.cpp 3629 0x54a707a8
8 QThread::启动 moc_qthread.cpp 160 0x54650149
9 QThreadPrivate::启动 qthread_win.cpp 377 0x5465cad6
10 BaseThreadInitThunk KERNEL32 0x7ffb9f 5b8364 11 RtlUserThreadStart ntdll 0x7ffba1bb7091