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請求時出現錯誤:\n網址:%1\n錯誤信息:%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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM