Qt的网络操作类是异步(非阻塞的),但有时想做一些阻塞的事情就不方便了,可用如下几行代码轻松实现:
1 QByteArray MyNetworkAccess::get(const QString &strUrl) 2 { 3 assert(!strUrl.isEmpty()); 4 5 const QUrl url = QUrl::fromUserInput(strUrl); 6 assert(url.isValid()); 7 8 QNetworkRequest qnr(url); 9 QNetworkReply* reply = m_qnam.get(qnr); //m_qnam是QNetworkAccessManager对象 10 11 QEventLoop eventLoop; 12 connect(reply, &QNetworkReply::finished, &eventLoop, &QEventLoop::quit); 13 eventLoop.exec(QEventLoop::ExcludeUserInputEvents); 14 15 QByteArray replyData = reply->readAll(); 16 reply->deleteLater(); 17 reply = nullptr; 18 19 return replyData; 20 }
当然如上方式不支持重定向(301等),因为暂时用不上,如果要支持,还要在return前判断并循环或递归。
另外如果出现error,上述方式会把服务器返回的错误信息直接返回,支持判断错误的版本请继续往下看!
并且可以看出本来是封装了一个网络操作类,但现在只有get,post等还没做,等弄好了也一起放上来!
第二版——支持判断error和重定向(error和重定向均按错误处理):
1 QByteArray MyNetworkAccess::get(const QString &strUrl) 2 { 3 assert(!strUrl.isEmpty()); 4 5 const QUrl url = QUrl::fromUserInput(strUrl); 6 assert(url.isValid()); 7 8 QNetworkRequest qnr(url); 9 QNetworkReply* reply = m_qnam.get(qnr); //m_qnam是QNetworkAccessManager对象 10 11 QEventLoop eventLoop; 12 connect(reply, &QNetworkReply::finished, &eventLoop, &QEventLoop::quit); 13 eventLoop.exec(QEventLoop::ExcludeUserInputEvents); 14 15 QByteArray replyData = reply->readAll(); 16 int statusCode = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt(); 17 QVariant redirectAttr = reply->attribute(QNetworkRequest::RedirectionTargetAttribute); 18 if (reply->error() 19 || 300 == statusCode //状态码300 Multiple Choices,既不是错误也不算重定向,应该是qt bug 20 || !redirectAttr.isNull()) 21 { 22 QString errString = reply->error() ? reply->errorString() : QString("发生重定向(%1),不允许此情况").arg(statusCode); 23 QMessageBox::critical(nullptr, "网络异常", 24 QString("发送get请求时出现错误: 网址:%1 错误信息:%2").arg(reply->request().url().toDisplayString(), errString)); 25 replyData.clear(); 26 } 27 28 reply->deleteLater(); 29 reply = nullptr; 30 31 return replyData; 32 }