QThread的和QTimer(QThread and QTimer)

2019-08-19 19:49发布

我正在使用Qt 4.6开发的应用程序。

我想创建一个单独的线程计算自定义计时器。 不过,我想这个计时器能够将信号发送到主线程。

我子类QThread的,但它似乎并没有工作。

这里是Timer.h:

#ifndef TIMER_H
#define TIMER_H

#include <QtCore/QObject>
#include <QtCore/QThread>
#include <QtCore/QTimer>

class Timer : public QThread
{
    Q_OBJECT
public:
    explicit Timer(QObject *parent = 0);
    ~Timer();

    // true if the timer is active
    bool isCounting();

    // start the timer with a number of seconds
    void startCounting(int value = 300);
    void stopCounting();

    // the number of seconds to reach
    int maximum();

    // the current value of the timer
    int value();

    // elapsed time since the timer has started
    int elapsedTime();

signals:
    // sent when the timer finishes to count
    void timeout();
    // an event is emited at each second when the timer is active
    void top(int remainingSeconds);

protected:
    // launch the thread
    //virtual void run();

private slots:
    // decrements the remaining time at each second and emits top()
    void timerEvent();

private:
    QTimer* _timer;
    // remaining time
    int _left;
    // number of seconds at timer startup
    int _maximum;
};

#endif // TIMER_H

和Timer.cpp:

#include "Timer.h"

Timer::Timer(QObject *parent) :
    QThread(parent)
{
    _timer = new QTimer(this);
    _maximum = 0;
    _left = 0;
    connect(_timer, SIGNAL(timeout()), this, SLOT(timerEvent()));
}

Timer::~Timer()
{
    delete _timer;
}

bool Timer::isCounting()
{
    // test if timer still active
    return _timer->isActive();
}

void Timer::startCounting(int value)
{
    qDebug() << QString("Start timer for %1 secs").arg(QString::number(value));
    if(_left != 0 || _timer->isActive())
    {
         _timer->stop();
    }

    _maximum = value;
    _left = value;

    // emit the first top
    emit top(_left);

    // start the timer: 1000 msecs
    _timer->start(1000);

    // start the thread
    start();
}

void Timer::stopCounting()
{
    qDebug() << QString("Stopping timer at %1 secs => %2 secs remaining.").arg(QString::number(elapsedTime()), QString::number(_left));
    // stop timer
    _timer->stop();
    _left = 0;
    _maximum = 0;
    // kill thread
    terminate();
}

int Timer::maximum()
{
    return _maximum;
}

int Timer::value()
{
    return _left;
}

void Timer::timerEvent()
{
    qDebug() << "Timer event";
    if(--_left == 0)
    {
        // stop timer
        _timer->stop();
        // emit end of timer
        emit timeout();
        // stop thread
        terminate();
    }
    else
    {
        // emit a signal at each second
        emit top(_left);
    }
}

int Timer::elapsedTime()
{
    return (_maximum - _left);
}

编辑

我意识到,我想移动到另一个线程的对象实际上是一个单例。 这可能会导致一个问题( 见这里 )。

Answer 1:

你不需要子类的QThread在这种特殊情况下。 而在一般情况下,从子类化QThread的,除非你确信这是你所需要的弃权。

下面是一个简单的例子如何建立一个工人和定时器在一个线程并启动它:

工人阶级:

class Worker : public QObject
{
    Q_OBJECT
public:
    explicit Worker(QObject *parent = 0) : QObject(parent) {}

signals:
    void doSomething();

public slots:
    void trigger() {
        emit doSomething();
    }
};

main.cpp中

int main(int argc, char *argv[])
{
    QCoreApplication a(argc, argv);

    MainThreadObject o;

    QThread *thread = new QThread;
    Worker w;
    QTimer timer;
    timer.setInterval(1000);

    timer.moveToThread(thread);
    w.moveToThread(thread);

    QObject::connect(thread, SIGNAL(started()), &timer, SLOT(start()));
    QObject::connect(&w, SIGNAL(doSomething()), &o, SLOT(doSomething()));
    QObject::connect(&timer, SIGNAL(timeout()), &w, SLOT(trigger()));

    thread->start();

    return a.exec();
}

所以,我们有MainThreadObject代表一个QObject的活在主线程。 我们创建计时器,并且Worker对象,它只是用于包装的信号和时隙,以避免子类的QThread的需要。 定时器设定完成且它和工人移动到新的线程,线程started()信号被连接到所述定时器start()槽,工人doSomething()信号被连接到主线程对象doSomething()插槽,最后计时器timeout()信号被连接到辅助trigger()插槽。 然后线程启动其在启动事件循环的整个链条。

其结果是,该MainThreadObject::doSomething()被调用每一秒,用来自辅助线程发出的信号。



Answer 2:

尝试

QMetaObject::invokeMethod(&timer, "start", Qt::QueuedConnection); //timer->start()

如果你想立即启动定时器

要么

QMetaObject::invokeMethod(&timer, "start", Qt::QueuedConnection , Q_ARG(int, 1000 )); //timer->start(200)

如果你想开始后,1000计时器

在非GUI螺纹(或的QThread回调的Pthread)



Answer 3:

首先,如果你从QThread的子类,你必须实现run()方法,如果没有,也没有这样做的时候,你可以继承QObject来代替。

其次,你的QTimer必须驻留在运行事件循环的线程。 如果没有事件循环没有Qt的队列信号可被发送。 您可以通过调用exec()在线程的run方法,启动事件循环:

void Timer::run() {
    exec();
}


Answer 4:

可能的原因可能是,你的计时器对象不与事件循环的线程。 事件循环需要触发信号。

不过,我建议你不要用这种方法去。 定时器使用不同的平台上不同的机制和不同的平台上按照预期的代码可能无法表现。



文章来源: QThread and QTimer