從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