从Qt4.4开始,引入了QNetworkRequest、QNetworkReply 和 QNetworkAccessManager等类来进行HTTP、FTP的操作,替代之前的QFtp和QHttp。很多情况下采用QNetworkAccessManager的finished信号构建异步方式。
1 //构建一个manager对象 2 QNetworkAccessManager *manager = new QNetworkAccessManager(this); 3 //manager具有异步API,当http请求完成后,会通过finished信号进行通知 4 connect(manager,&QNetworkAccessManager::finished,this,&MyClass::replyFinished); 5 //发送异步get请求 6 manager->get(QNetworkRequest(QUrl("http://qt-project.org"))); 7 8 //这里也可以用一个QEventLoop来等待请求完成,但是我更爱用槽函数 9 //QNetworkReply *reply=manager->get(request); 10 //QEventLoop eventLoop; 11 //connect(manager, &QNetworkAccessManager::finished, &eventLoop, &QEventLoop::quit); 12 //eventLoop.exec(); 13 //QByteArray reply_data=reply->readAll();
Qt的网络操作类是异步(非阻塞的),但有时想做一些阻塞的事情就不方便了,特别是需要在当前函数中直接获得返回值时,则可以使用QEventLoop阻塞运行,上面的注释部分。具体的用例如下代码所示:
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 }
注意:以下方式可不行哦
1、通过QNetworkReply中的isFinished()或isRunning函数,在while循环中判断reply是否已经结束——不可行
转自:https://www.cnblogs.com/roadbike/p/6055263.html