QEventLoop 没有同步等待 QNetworkReply 完成

Posted

技术标签:

【中文标题】QEventLoop 没有同步等待 QNetworkReply 完成【英文标题】:QEventLoop not waiting synchronously for QNetworkReply to finish 【发布时间】:2021-04-10 08:04:59 【问题描述】:

我正在使用 Qt 构建一个调用服务器的库,我需要构建一个等待 HTTP 响应(QNetworkReply 对象)的同步函数,并且我正在使用 QEventLoop 来实现这一点。目前服务器被调用,但循环不等待回复完成,而是继续使用一个空的 QNetworkReply 对象。

完全相同的函数在我构建的一个简单测试项目中工作,该项目仅包含一个线程和一个从 main 到该函数的调用。等待回复,一切都按预期工作。但是在我的包含多个线程的项目中,会发生上述场景并且事件循环不等待回复。请求被发送到服务器并显示在那里,但响应无法返回到 QNetworkReply 对象,因为该函数已经执行。

这是我的功能的网络部分。在我的项目中,statusCode 变量始终为 0,回复为空,但在简单的测试场景中,它们为 200 和预期的 HTTP 响应。

QNetworkAccessManager* networkManager = new QNetworkAccessManager(this);
QNetworkReply* reply = networkManager->get(request);

QEventLoop loop;
connect(reply, SIGNAL(finished()), &loop, SLOT(quit()));
loop.exec();

QVariant statusCode = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute);
qInfo() << "Http Get completed with status code:" << statusCode.toInt();

【问题讨论】:

您声明"...works in a simple test project I built that only contains one thread"。如果正在使用多个线程,那么您需要提供更多关于如何创建/管理这些线程的上下文——最好是minimal reproducible example。 【参考方案1】:

原因

对不起,我没听你的话……

换句话说,您在发出请求之后连接到finished 信号,因此在发出请求时没有人在监听您的信号。

解决方案

连接到QNetworkAccessManager::finished您使用networkManager-&gt;get(request)发送请求之前。

示例

请仔细阅读QNetworkAccessManager的详细说明。有如何正确使用该类的示例。例如:

QNetworkAccessManager *manager = new QNetworkAccessManager(this);
connect(manager, &QNetworkAccessManager::finished,
        this, &MyClass::replyFinished);

manager->get(QNetworkRequest(QUrl("http://qt-project.org")));

【讨论】:

【参考方案2】:
QNetworkAccessManager *manager = new QNetworkAccessManager(this);
QUrl resource(url);
QNetworkRequest request(resource);
QNetworkReply *reply = manager->get(request);
QEventLoop loop;
connect(reply, SIGNAL(finished()), &loop, SLOT(quit()));
loop.exec();
QJsonObject jsonObject = QJsonDocument::fromJson(reply->readAll()).object();

【讨论】:

以上是关于QEventLoop 没有同步等待 QNetworkReply 完成的主要内容,如果未能解决你的问题,请参考以下文章

QEventLoop 仅等待本地事件而不是主循环事件

QEventLoop等待另外一个事件的停止,非常实用 good

QWebEngine:同步执行 runJavascript - QEventLoop 阻止 Javascript 调用

本地 QEventLoop 是如何工作的

如何修复“在 Qt 中将两个定时器变为一个函数,使用 qmutex 将 qeventloop 进行睡眠”

QEventLoop 的使用两例