Qt实现同步(阻塞式)http get等网络访问操作

从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

原文地址:https://www.cnblogs.com/liushui-sky/p/13814416.html