Qt之處理QNetworkAccessManager網(wǎng)絡連接超時
簡述
在網(wǎng)絡操作中,經(jīng)常會由于各種原因引起網(wǎng)絡連接超時,究竟何為網(wǎng)絡連接超時?
網(wǎng)絡連接超時:在程序默認的等待時間內(nèi)沒有得到服務器的響應
簡述超時原因Qt 中的網(wǎng)絡連接超時如何處理超時封裝類超時原因
引起網(wǎng)絡連接超時的原因很多,下面,列舉一些常見的原因:
網(wǎng)絡斷開,不過經(jīng)常顯示無法連接網(wǎng)絡阻塞,導致你不能在程序默認等待時間內(nèi)得到回復數(shù)據(jù)包網(wǎng)絡不穩(wěn)定,網(wǎng)絡無法完整傳送服務器信息系統(tǒng)問題,系統(tǒng)資源過低,無法為程序提供足夠的資源處理服務器信息設備不穩(wěn)定,如網(wǎng)線松動、接口沒插好等等網(wǎng)絡注冊時系統(tǒng)繁忙,無法回應網(wǎng)速過慢,如 使用 BT 多線程下載,在線收看視頻等大量占用帶寬的軟件 ,若使用共享帶寬還要防范他人惡意占用帶寬計算機感染了惡意軟件,計算機病毒,計算機木馬等Qt 中的網(wǎng)絡連接超時
在 Qt 中,關于 QNetworkAccessManager、QNetworkRequest 和 QNetworkReply 的文檔中,找到了有關超時相關的錯誤 QNetworkReply::NetworkError。
常量 QNetworkReply::TimeoutError:
the connection to the remote server timed out
瞬間欣喜若狂,既然有超時錯誤,必然有設置超時的接口吧!遺憾,遺憾,遺憾。。。重要的事情說 3 遍,翻遍了官方文檔,能和超時扯上關系的就這么一個簡單的常量說明(當然還有 QNetworkReply::ProxyTimeoutError)。
這種情況下,我們只能自己去處理超時了。
如何處理超時
解決思路:
使用 QTimer 啟動一個單次定時器,并設置超時時間。在事件循環(huán)退出之后,判斷定時器的狀態(tài),如果是激活狀態(tài),證明請求已經(jīng)完成;否則,說明超時。
來看一個簡單的例子 - 獲取?Qt 官網(wǎng)?網(wǎng)頁內(nèi)容:
QTimer?timer; timer.setInterval(30000);??//?設置超時時間?30?秒 timer.setSingleShot(true);??//?單次觸發(fā) //?請求?Qt?官網(wǎng) QNetworkAccessManager?manager; QNetworkRequest?request; request.setUrl(QUrl("http://qt-project.org")); request.setRawHeader("User-Agent",?"MyOwnBrowser?1.0"); QNetworkReply?*pReply?=?manager.get(request); QEventLoop?loop; connect(&timer,?&QTimer::timeout,?&loop,?&QEventLoop::quit); connect(pReply,?&QNetworkReply::finished,?&loop,?&QEventLoop::quit); timer.start(); loop.exec();??//?啟動事件循環(huán) if?(timer.isActive())?{??//?處理響應 ????timer.stop(); ????if?(pReply->error()?!=?QNetworkReply::NoError)?{ ????????//?錯誤處理 ????????qDebug()?<<?"Error?String?:?"?<<?pReply->errorString(); ????}?else?{ ????????QVariant?variant?=?pReply->attribute(QNetworkRequest::HttpStatusCodeAttribute); ????????int?nStatusCode?=?variant.toInt(); ????????//?根據(jù)狀態(tài)碼做進一步數(shù)據(jù)處理 ????????//QByteArray?bytes?=?pReply->readAll(); ????????qDebug()?<<?"Status?Code?:?"?<<?nStatusCode; ????} }?else?{??//?處理超時 ????disconnect(pReply,?&QNetworkReply::finished,?&loop,?&QEventLoop::quit); ????pReply->abort(); ????pReply->deleteLater(); ????qDebug()?<<?"Timeout"; }
首先,定義一個 QTimer,設置超時時間為 30000 毫秒(30 秒)并設置為單次觸發(fā)。然后,使用 QNetworkRequest 實現(xiàn)一個簡單的網(wǎng)絡請求,通過 QNetworkAccessManager::get() 開始獲取 Qt 官網(wǎng)的 HTML 頁面內(nèi)容。因為請求過程是異步的,所以通過使用 QEventLoop 啟動一個事件循環(huán)讓其同步處理,并將 QTimer 的 timeout() 信號以及 QNetworkReply 的 finished() 信號連接至其 quit() 槽函數(shù),保證在定時器過期之后或者網(wǎng)絡響應完成后事件循環(huán)得到退出,不至于一直處于阻塞狀態(tài)。
如上所述,事件循環(huán)退出的兩種情況:
QTimer 30 秒到期,超時網(wǎng)絡連接響應完成
所以,當 QTimer::isActive() 激活的情況下,證明響應完成,還尚未超時。這時需要先調(diào)用 QTimer::stop() 來停止定時器,再對響做進一步處理。否則,進行超時處理 - QNetworkReply::abort() 立即中止操作并關閉網(wǎng)絡連接。
封裝類
既然以后會經(jīng)常用到,那么還是提供一個封裝類 QReplyTimeout 專門處理超時。
#include#include#includeclass?QReplyTimeout?:?public?QObject?{ ????Q_OBJECT public: ????QReplyTimeout(QNetworkReply?*reply,?const?int?timeout)?:?QObject(reply)?{ ????????Q_ASSERT(reply); ????????if?(reply?&&?reply->isRunning())?{??//?啟動單次定時器 ????????????QTimer::singleShot(timeout,?this,?SLOT(onTimeout())); ????????} ????} signals: ????void?timeout();??//?超時信號?-?供進一步處理 private?slots: ????void?onTimeout()?{??//?處理超時 ????????QNetworkReply?*reply?=?static_cast(parent()); ????????if?(reply->isRunning())?{ ????????????reply->abort(); ????????????reply->deleteLater(); ????????????emit?timeout(); ????????} ????} };
由于 QNetworkReply 和 QReplyTimeout 是父子關系,所以 QReplyTimeout 將被自動銷毀。
使用起來非常簡單:
QNetworkAccessManager?*pManger?=?new?QNetworkAccessManager(this); QNetworkReply?*pReply?=?pManger->get(QNetworkRequest(QUrl("https://www.google.com"))); QReplyTimeout?*pTimeout?=?new?QReplyTimeout(pReply,?1000); //?超時進一步處理 connect(pTimeout,?&QReplyTimeout::timeout,?[=]()?{ ????qDebug()?<<?"Timeout"; });
如果對 Google 的獲取未在 1000 毫秒(1 秒)內(nèi)完成,則會中止,并發(fā)出 timeout() 信號,供進一步處理(例如:提示用戶請求超時)。