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

QThread emits finished() signal but isRunning() returns true and isFinished() returns false

本文关键字:返回 true isFinished fals 发出 isRunning 信号 QThread finished      更新时间:2023-10-16

下面是我的qthread实现的代码。我想从卫星上获取gps数据。即使程序退出gpsSearch()插槽函数,QThread也不会产生finished()信号。每当单击按钮时,就调用locateMe()函数。当线程未启动时,第一次点击按钮,isRunning()函数打印真值,isFinished()函数打印假值。我必须调用QTherad的quit()函数来手动停止线程。之后,它转到gnssProvider类中连接的threadQuit()函数。但即使在那之后,如果我点击按钮,它打印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();
 }

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. 线程实际终止

所以你需要调用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()插槽。

这是预期的行为,线程不会退出,直到您手动终止它。线程对象承载一个事件循环,所以直到事件循环退出,它才会完成,正如Sebastian所解释的。

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

您使用哪个Qt版本?

Qt 4.8返回错误的值,直到4.8.4 (Qt bug 30251)。