如何在Qt中进行“死人切换”?

时间:2019-01-24 08:34:48

标签: qt timeout qt5 qtimer

我想监视一个定期进行的操作,以便在停止时可以执行某些操作。例如,切换到本地计时器以继续执行该操作,然后在再次启动时切换回主要来源。

我能想到的最好的理论例子是

#define FRAME_INTERVAL_MSEC     33  // ~30Hz
bool periodFromUSB;
QTimer* myTimer;

void Processing::localTimeout()
{
    //connected to myTimer->timeout();
    if (periodFromUSB)
    {
        periodFromUSB = false;
        myTimer->stop();
        myTimer->setSingleShot(false);
        myTimer->setInterval(FRAME_INTERVAL_MSEC);
        myTimer->start();
    }

    processDMX();
}

void Processing::newFrame()
{
    //called periodically from my USB driver
    myTimer->stop();
    if(!periodFromUSB)
    {
        periodFromUSB = true;
        myTimer->setSingleShot(true);
        myTimer->setInterval(FRAME_INTERVAL_MSEC * 2);
    }
    myTimer->start();

    processDMX();
}

但是无论我如何安排,当我的USB驱动程序运行时,我都会不断调用这两个函数。 (上面代码中任何地方的断点都会立即显示)

有没有更好的做死人的方法?

1 个答案:

答案 0 :(得分:1)

如果调用start(),计时器将重置为0并重新启动当前间隔。也就是说,只要在达到超时之前调用newFrame()插槽,就永远不会调用localTimeout()。

您的代码将如下所示:

#define FRAME_INTERVAL_MSEC     33  // ~30Hz
bool periodFromUSB;
QTimer* myTimer;

void Processing::localTimeout()
{
    //connected to myTimer->timeout();
    if (periodFromUSB)
    {
        periodFromUSB = false;
    }

    processDMX();
}

void Processing::newFrame()
{
    //called periodically from my USB driver
    if(!periodFromUSB)
    {
        periodFromUSB = true;
    }
    myTimer->start(); // restarts current timer interval

    processDMX();

    // for debugging: display actual interval length in milliseconds:
    static qint64 start = QDateTime::currentMSecsSinceEpoch();
    static qint64 count = 0;
    ++count;
    qint64 curr = QDateTime::currentMSecsSinceEpoch();
    float msecsPerInterval = float(curr - start) / float(count);
    qDebug() << "msecs per interval: " << msecsPerInterval;
}

void Processing::init()
{
    myTimer = new QTimer();
    connect(myTimer, &QTimer::timeout, this, &Processing::newFrame);
    myTimer->setInterval(FRAME_INTERVAL_MSEC);
    myTimer->start();
}

如果您的localTimeout()插槽仍被调用,则您的FRAME_INTERVAL_MSEC太低。

您可以使用我添加到newFrame()的调试代码进行检查。

这可能是因为:

  • 您的USB驱动程序很少接收数据或
  • processDMX()中的处理时间太长(如果processDMX()的持续时间长于FRAME_INTERVAL_MSEC,则不会及时调用下一个时隙)