QSignalSpy нельзя использовать с потоками

Я написал поток, который выполняет рабочий объект. Все отлично работает Также результирующие сигналы излучаются так, как должны. Конечно, я позаботился об обычных ошибках в отношении схожести потоков / объектов.

Сегодня я написал автоматизированный модульный тест для этих рабочих / потоков. Я создал QSignalSpy для ожидания сигнала, который испускается рабочим объектом (который был перемещен в поток), например так:

QSignalSpy spy(worker, SIGNAL(Success()));
thread.ExecuteWorker();
QVERIFY(spy.wait()); // Error in this line

Я получаю известную ошибку в отмеченной строке:

QObject::killTimer: timers cannot be stopped from another thread

Сначала я ожидал ошибку на моей стороне, потому что некоторый код в wait() был выполнен не в том потоке. Затем я нашел следующий код в реализации QSignalSpy:

if (!QMetaObject::connect(obj, sigIndex, this, memberOffset, Qt::DirectConnection, 0))
{
   qWarning("QSignalSpy: QMetaObject::connect returned false. Unable to connect.");
   return;
}

Это, очевидно, означает, что QSignalSpy использует DirectConnection все время и не может использоваться для мониторинга сигналов объектов, находящихся в разных потоках.

Почему они так программировали в Qt5.3? Это ошибка или намеренное поведение? Как я могу обойти это ограничение?

2 ответа

Решение

К сожалению, это давняя проблема, более шести лет, чтобы быть справедливой:

QSignalSpy падает, если сигнал испускается из рабочего потока

Я встретил Джейсона на саммите Qt Contributor пару лет назад, но вскоре после этого он покинул Nokia, когда Nokia закрыла офис в Брисбене, где он работал. После этого, к сожалению, в этом тестовом модуле Qt не было особого вклада.

Недавно в списке рассылки также было больше дискуссий об этом:

Почему QSignalSpy использует Qt::DirectConnection?

Решение, предложенное Роландом, заключалось в том, что сопровождающий, Тиаго, также принял:

if (thread() != QThread::currentThread())
{
    QMetaObject::invokeMethod(this, "exitLoop", Qt::QueuedConnection);
    return;
}

Немного стыдно, что это не до 5.4. Сказав это, это будет исправлено в Qt 5.4, поскольку изменение было объединено:

Сделать QTestEventLoop::exitLoop() потокобезопасным

Чтобы обеспечить надежную работу QSignalSpy в потоках, я использую следующий подход: я перемещаю шпиона в рабочий поток и заново реализую функцию ожидания следующим образом:

#include <QSignalSpy>
#include <QTime>
struct ThreadsafeQSignalSpy : QSignalSpy
{
    template <typename Func>
    ThreadsafeQSignalSpy(const typename QtPrivate::FunctionPointer<Func>::Object *obj, Func signal0)
        : QSignalSpy(obj, signal0)
    {}

    bool wait(int timeout)
    {
        auto origCount(count());
        QTime timer;
        timer.start();

        while (count() <= origCount && timer.elapsed() < timeout)
            QCoreApplication::instance()->processEvents(QEventLoop::AllEvents, timeout/10);
        return count() > origCount;
    }
};


void TestSuite::testFunction()
{
    QThread thread;
    ...
    ThreadsafeQSignalSpy spy;
    spy.moveToThread(thread);
    /// now wait should work
    ...
    QVERIFY(spy.wait(1000));
}
Другие вопросы по тегам