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請求時出現錯誤:\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 }