Qt之處理QNetworkAccessManager網(wǎng)絡(luò)連接超時(shí)
簡(jiǎn)述
在網(wǎng)絡(luò)操作中,經(jīng)常會(huì)由于各種原因引起網(wǎng)絡(luò)連接超時(shí),究竟何為網(wǎng)絡(luò)連接超時(shí)?
網(wǎng)絡(luò)連接超時(shí):在程序默認(rèn)的等待時(shí)間內(nèi)沒(méi)有得到服務(wù)器的響應(yīng)
簡(jiǎn)述超時(shí)原因Qt 中的網(wǎng)絡(luò)連接超時(shí)如何處理超時(shí)封裝類(lèi)超時(shí)原因
引起網(wǎng)絡(luò)連接超時(shí)的原因很多,下面,列舉一些常見(jiàn)的原因:
網(wǎng)絡(luò)斷開(kāi),不過(guò)經(jīng)常顯示無(wú)法連接網(wǎng)絡(luò)阻塞,導(dǎo)致你不能在程序默認(rèn)等待時(shí)間內(nèi)得到回復(fù)數(shù)據(jù)包網(wǎng)絡(luò)不穩(wěn)定,網(wǎng)絡(luò)無(wú)法完整傳送服務(wù)器信息系統(tǒng)問(wèn)題,系統(tǒng)資源過(guò)低,無(wú)法為程序提供足夠的資源處理服務(wù)器信息設(shè)備不穩(wěn)定,如網(wǎng)線(xiàn)松動(dòng)、接口沒(méi)插好等等網(wǎng)絡(luò)注冊(cè)時(shí)系統(tǒng)繁忙,無(wú)法回應(yīng)網(wǎng)速過(guò)慢,如 使用 BT 多線(xiàn)程下載,在線(xiàn)收看視頻等大量占用帶寬的軟件 ,若使用共享帶寬還要防范他人惡意占用帶寬計(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í)的接口吧!遺憾,遺憾,遺憾。。。重要的事情說(shuō) 3 遍,翻遍了官方文檔,能和超時(shí)扯上關(guān)系的就這么一個(gè)簡(jiǎn)單的常量說(shuō)明(當(dāng)然還有 QNetworkReply::ProxyTimeoutError)。
這種情況下,我們只能自己去處理超時(shí)了。
如何處理超時(shí)
解決思路:
使用 QTimer 啟動(dòng)一個(gè)單次定時(shí)器,并設(shè)置超時(shí)時(shí)間。在事件循環(huán)退出之后,判斷定時(shí)器的狀態(tài),如果是激活狀態(tài),證明請(qǐng)求已經(jīng)完成;否則,說(shuō)明超時(shí)。
來(lái)看一個(gè)簡(jiǎn)單的例子 - 獲取?Qt 官網(wǎng)?網(wǎng)頁(yè)內(nèi)容:
QTimer?timer;
timer.setInterval(30000);??//?設(shè)置超時(shí)時(shí)間?30?秒
timer.setSingleShot(true);??//?單次觸發(fā)
//?請(qǐng)求?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è)簡(jiǎn)單的網(wǎng)絡(luò)請(qǐng)求,通過(guò) QNetworkAccessManager::get() 開(kāi)始獲取 Qt 官網(wǎng)的 HTML 頁(yè)面內(nèi)容。因?yàn)檎?qǐng)求過(guò)程是異步的,所以通過(guò)使用 QEventLoop 啟動(dòng)一個(gè)事件循環(huán)讓其同步處理,并將 QTimer 的 timeout() 信號(hào)以及 QNetworkReply 的 finished() 信號(hào)連接至其 quit() 槽函數(shù),保證在定時(shí)器過(guò)期之后或者網(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() 來(lái)停止定時(shí)器,再對(duì)響做進(jìn)一步處理。否則,進(jìn)行超時(shí)處理 - QNetworkReply::abort() 立即中止操作并關(guān)閉網(wǎng)絡(luò)連接。
封裝類(lèi)
既然以后會(huì)經(jīng)常用到,那么還是提供一個(gè)封裝類(lèi) QReplyTimeout 專(zhuān)門(mén)處理超時(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)銷(xiāo)毀。
使用起來(lái)非常簡(jiǎn)單:
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";
});如果對(duì) Google 的獲取未在 1000 毫秒(1 秒)內(nèi)完成,則會(huì)中止,并發(fā)出 timeout() 信號(hào),供進(jìn)一步處理(例如:提示用戶(hù)請(qǐng)求超時(shí))。





