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->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等待另外一个事件的停止,非常实用 good
QWebEngine:同步执行 runJavascript - QEventLoop 阻止 Javascript 调用