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