关闭 QCoreApplication 的正确方法
Proper way to close QCoreApplication
我正在制作一个 Qt5 QCoreApplication 用于从 FTP 服务器下载文件(从 HTTP 开始,现在已切换)。
我的程序要关闭时遇到问题。在我将 exit(0)
添加到 downloader.cpp
之后,我的程序现在结束了,但是我得到以下错误:
QWaitCondition: Destroyed while threads are still waiting.
我的代码如下:
main.cpp
#include <QCoreApplication>
#include <downloader.h>
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Downloader d;
d.doDownload();
a.exec();
}
**downloader.cpp**
#include "downloader.h"
Downloader::Downloader(QObject *parent) :
QObject(parent)
{
}
void Downloader::doDownload() {
manager = new QNetworkAccessManager(this);
connect(manager, SIGNAL(finished(QNetworkReply*)), this, SLOT(replyFinished(QNetworkReply*)));
manager->get(QNetworkRequest(QUrl("ftp://ftp.fao.org/Public/GIEWS/windisp/40manual/wd4en.pdf")));
}
void Downloader::replyFinished (QNetworkReply *reply)
{
if(reply->error()) {
qDebug() << "ERROR!";
qDebug() << reply->errorString();
}
else
{
qDebug() << "Download finished!";
QFile *file = new QFile("C:/Users/jelicicm/Desktop/wd4en.pdf");
if(file->open(QFile::Append))
{
file->write(reply->readAll());
file->flush(); file->close();
qDebug() <<"Downloaded file size:" <<(file->size())/1024<<"KB";
}
delete file;
}
reply->deleteLater();
exit(0);
}
我得到以下输出:
Download Finished! Downloaded file size... QWaitCondition: Destroyed
while threads are still waiting.
据我所见,我想象的一切都实现了。下载文件,并显示其大小。但我想这个错误一定意味着什么。
谁能给我解释一下这个错误是什么,为什么会发生以及如何修补它?
您的代码有一些问题。您将 Qt 视为本质上是过程性的,而实际上它是通过主循环由事件驱动的。
首先,在QApplication
循环开始之前调用qApp->exit(0)
是错误的。根据 Qt,“如果事件循环不是 运行,这个函数什么都不做。” 你的事件循环还不是 运行 因为你调用了 doDownload
在你打电话给 exec
之前
其次,到您 运行 QApplication::exec
时,您还没有创建任何顶级 windows 或要调度的事件。我不确定当您无事可做地调用 exec
时应该发生什么。当然,从语义上讲,对 exec
的调用什么也不做。从技术上讲……也许它会导致线程错误。同样,我不确定,但我知道您不应该以这种方式使用 exec
。
运行 你的主事件循环 before 做所有你喜欢的 Qt 东西。这允许 Qt 调用您的代码。添加一个插槽 doDownload
和一个信号 finished
到您的 Downloader
.
class Downloader : public QObject
{
Q_OBJECT
public:
Downloader(QObject * parent = nullptr);
private slots:
void doDownload();
signals:
void finished();
}
...
void Downloader::doDownload()
{
// Same implementation as before
// Emit signal when finished
emit finished();
}
然后在通过调用 exec
并启动主循环建立控制反转后调用您的插槽:
#include <QCoreApplication>
#include "Downloader.h"
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Downloader d;
// Quit application when work is finished
QObject::connect(&d, SIGNAL(finished()), &a, SLOT(quit())); // changed the
//variable name 'app' to 'a'
// Run the user-hook (doDownload) from the application event loop.
QTimer::singleShot(0, &d, SLOT(doDownload()));
return a.exec();
}
现在 Qt 将调用您的代码。无需显式退出应用程序,只需 emit finished()
即可正确清理所有内容。
如果这能解决您的问题,请告诉我。如果没有,可能还有其他鱼要炸。
我正在制作一个 Qt5 QCoreApplication 用于从 FTP 服务器下载文件(从 HTTP 开始,现在已切换)。
我的程序要关闭时遇到问题。在我将 exit(0)
添加到 downloader.cpp
之后,我的程序现在结束了,但是我得到以下错误:
QWaitCondition: Destroyed while threads are still waiting.
我的代码如下:
main.cpp
#include <QCoreApplication>
#include <downloader.h>
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Downloader d;
d.doDownload();
a.exec();
}
**downloader.cpp**
#include "downloader.h"
Downloader::Downloader(QObject *parent) :
QObject(parent)
{
}
void Downloader::doDownload() {
manager = new QNetworkAccessManager(this);
connect(manager, SIGNAL(finished(QNetworkReply*)), this, SLOT(replyFinished(QNetworkReply*)));
manager->get(QNetworkRequest(QUrl("ftp://ftp.fao.org/Public/GIEWS/windisp/40manual/wd4en.pdf")));
}
void Downloader::replyFinished (QNetworkReply *reply)
{
if(reply->error()) {
qDebug() << "ERROR!";
qDebug() << reply->errorString();
}
else
{
qDebug() << "Download finished!";
QFile *file = new QFile("C:/Users/jelicicm/Desktop/wd4en.pdf");
if(file->open(QFile::Append))
{
file->write(reply->readAll());
file->flush(); file->close();
qDebug() <<"Downloaded file size:" <<(file->size())/1024<<"KB";
}
delete file;
}
reply->deleteLater();
exit(0);
}
我得到以下输出:
Download Finished! Downloaded file size... QWaitCondition: Destroyed while threads are still waiting.
据我所见,我想象的一切都实现了。下载文件,并显示其大小。但我想这个错误一定意味着什么。
谁能给我解释一下这个错误是什么,为什么会发生以及如何修补它?
您的代码有一些问题。您将 Qt 视为本质上是过程性的,而实际上它是通过主循环由事件驱动的。
首先,在QApplication
循环开始之前调用qApp->exit(0)
是错误的。根据 Qt,“如果事件循环不是 运行,这个函数什么都不做。” 你的事件循环还不是 运行 因为你调用了 doDownload
在你打电话给 exec
其次,到您 运行 QApplication::exec
时,您还没有创建任何顶级 windows 或要调度的事件。我不确定当您无事可做地调用 exec
时应该发生什么。当然,从语义上讲,对 exec
的调用什么也不做。从技术上讲……也许它会导致线程错误。同样,我不确定,但我知道您不应该以这种方式使用 exec
。
运行 你的主事件循环 before 做所有你喜欢的 Qt 东西。这允许 Qt 调用您的代码。添加一个插槽 doDownload
和一个信号 finished
到您的 Downloader
.
class Downloader : public QObject
{
Q_OBJECT
public:
Downloader(QObject * parent = nullptr);
private slots:
void doDownload();
signals:
void finished();
}
...
void Downloader::doDownload()
{
// Same implementation as before
// Emit signal when finished
emit finished();
}
然后在通过调用 exec
并启动主循环建立控制反转后调用您的插槽:
#include <QCoreApplication>
#include "Downloader.h"
int main(int argc, char *argv[])
{
QCoreApplication a(argc, argv);
Downloader d;
// Quit application when work is finished
QObject::connect(&d, SIGNAL(finished()), &a, SLOT(quit())); // changed the
//variable name 'app' to 'a'
// Run the user-hook (doDownload) from the application event loop.
QTimer::singleShot(0, &d, SLOT(doDownload()));
return a.exec();
}
现在 Qt 将调用您的代码。无需显式退出应用程序,只需 emit finished()
即可正确清理所有内容。
如果这能解决您的问题,请告诉我。如果没有,可能还有其他鱼要炸。