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





