1

我有一个使用一个工作线程的简单应用程序。这个工作线程启动并初始化DownloadManager,它负责从网络下载文件。在我的主应用程序类中,我在 DownloadManager 完成之前发出的线程上有了 finished() SIGNAL。

我的问题是如何让工作线程等到 DownloadManager 完成它的工作。

这是示例代码:

 class Main
 {
     m_DownloadWorker = new DownloadWorker(this);
     QObject::connect(pm_hotosDownloadWorker, SIGNAL(finished()), this, SLOT(DownloadWorkerFinished()));
     m_DownloadWorker->Execute();
     // do i need to do here something so the thread will wait ?

     .....


    void Main::DownloadWorkerFinished()
    {
        Log("DownloadWorkerFinished");
    }
 };


 class DownloadWorker : public QThread
 {
     void DownloadWorker::Execute()
     {

         // do i need to do here somthing so the thread will wait ? 
        start();
     }

     void DownloadWorker::run()
     {
         // do i need to do here somthing so the thread will wait ?
        DownloadManager* pDownloadManager = new DownloadManager(this);
        pDownloadManager->download();


     }
 };

 class DownloadManager: public QObject
 {
     // downloading stuff using Qt networkmanager 

     .....
     .....
 }
4

2 回答 2

5

如果在异步操作完成时发出信号,则始终可以使用 QEventLoop 将异步操作“转换”为与当前线程同步。它仍然是异步的,但线程将“等待”它完成。

QNetworkAccessManager nam;
QEventLoop loop;
QNetworkReply *reply = nam.get(request); 
connect(reply, SIGNAL(finished()), &loop, SLOT(quit()));
loop.exec();

所以基本上你把它放在你想要同步进行网络请求的 DownloadManager 类中。exec()一旦循环的quit槽被调用,就会返回。

于 2011-04-10T09:18:58.020 回答
3

您可以使用QThread::exec()调用在事件循环中运行您的线程。线程将运行它,直到您通过调用QThread::exit()告诉您的线程退出。所以一些示例代码可能如下所示:

void DownloadWorker::run()
 {
    DownloadManager* pDownloadManager = new DownloadManager(this);
    connect(pDownloadManager, SIGNAL(finished()), SLOT(exit()));
    connect(pDownloadManager, SIGNAL(error()), SLOT(exit()));
    pDownloadManager->download();
    exec();
 }

这将保证您的线程在您的 DownloadManager 的“finished()”信号发出之前不会退出。

注意:这里我举了一个如何解决您的问题的示例,但我不知道您的整个应用程序代码。这意味着不能保证此代码是线程安全且一致的。您需要自己处理互斥锁和所有正确的同步。要非常小心 !使用这种“低级”线程 API 需要对多线程有很好的理解。

希望有帮助

于 2011-04-10T18:29:13.900 回答