QThread发出finished()信号,但isRunning()返回true,isFinished()返回false

时间:2013-06-11 13:30:13

标签: c++ qt gps blackberry-10 qthread

以下是我的qthread实现的代码。我试图从卫星获取gps数据。即使程序退出gpsSearch()插槽功能,QThread也不会产生finished()信号。只要单击一个按钮,就会调用函数locateMe()。第一次没有启动线程并单击该按钮时,它会为isRunning()函数打印true值,并为isFinished()函数打印false值。我不得不调用QTherad的quit()函数来手动停止线程。之后,它将转到threadQuit()类中的连接gnssProvider函数。但即便如此,如果我单击按钮,它会在isRunning函数中为isFinished()locateMe()打印真值。

GPSInfo::GPSInfo()
{
    hybridGPSFound = satelliteGPSFound = networkGPSFound = false;
    qDebug()<<"Thread Creating";
    gnssThread = new QThread;
    gnssProvider = new LocationFetcher(this,GEOLOCATION_PROVIDER_GNSS,1);
    gnssProvider->moveToThread(gnssThread);
    connect(gnssThread, SIGNAL(started()), gnssProvider, SLOT(gpsSearch()));
    connect(gnssThread, SIGNAL(finished()), gnssProvider, SLOT(threadQuit()));
}
void LocationFetcher::gpsSearch()
{
    if (BPS_SUCCESS != geolocation_request_events(0))
    {
       fprintf(stderr, "Error requesting geolocation events: %s", strerror(errno));
       return;
    }
    geolocation_set_provider(GPS_Search_Provider);
    geolocation_set_period(GPS_Search_Period);
    while (!stopThread)
    {
        bps_event_t *event = NULL;
        bps_get_event(&event, -1);

        if (event)
        {
            if (bps_event_get_domain(event) == geolocation_get_domain() && bps_event_get_code(event) == GEOLOCATION_INFO)
            {
                handle_geolocation_response(event);
                break;
            }
        }
    }
    geolocation_stop_events(0);

    this->quit();

}
void GPSInfo::LocateMe()
{
    qDebug()<<"Thread Running: "<<gnssThread->isFinished();
    qDebug()<<"Thread Running: "<<gnssThread->isRunning();

    gnssThread->start();
    hybridThread->start();
    networkThread->start();

 }

3 个答案:

答案 0 :(得分:29)

QThread生命周期的工作方式如下:

  1. 您致电QThread::start()
  2. 此时,isRunning()应该开始返回true。
  3. 线程内部开始。它们发出started()信号。
  4. 主题内部呼叫run()
  5. 除非您在子类中覆盖此内容,否则run()会调用exec()
  6. exec()进入事件循环并一直待到quit()exit()
  7. exec()run()返回内部。
  8. 此时,isFinished()应该开始返回true并isRunning() false。
  9. 内部发出finished()信号。
  10. 内部进行最后的清理工作。
  11. 线程终止为真实。
  12. 因此,您需要在位置提取程序完成后调用quit() - 但this->quit()未在线程上调用quit()!这可能就是它没有做任何事情的原因。

    您的代码看起来有点像本文后的图案:

    http://mayaposch.wordpress.com/2011/11/01/how-to-really-truly-use-qthreads-the-full-explanation/

    请注意她如何向其工作人员发送finished()信号(与QThread::finished不同)并将其连接到QThread::quit()广告位。

答案 1 :(得分:1)

在您手动终止线程之前,线程不会退出。线程对象承载一个事件循环,因此在事件循环退出之前它不会完成,正如塞巴斯蒂安解释的那样。

简而言之,您的信号槽连接在概念上是向后的 - 对象应该在线程完成时终止线程,而不是相反。

答案 2 :(得分:0)

你使用哪个Qt版本?

Qt 4.8返回错误的值直到4.8.4(Qt bug 30251)。 此错误已在4.8.5中修复。