關(guān)于OkHttp的超時(shí)時(shí)間

雖然網(wǎng)上有很多關(guān)于okhttp超時(shí)時(shí)間的文章但大多都一筆帶過并沒有進(jìn)行詳細(xì)的講解各自的作用,于是就看了下源碼大致寫一下其中的發(fā)現(xiàn).
本文以 'com.squareup.okhttp3:okhttp:3.12.0'源碼為參考
首先我們一共可以設(shè)置5個(gè)超時(shí)時(shí)間分別如下:
OkHttpClient client = new OkHttpClient.Builder()
.connectTimeout(30, TimeUnit.SECONDS)
.callTimeout(120, TimeUnit.SECONDS)
.pingInterval(5, TimeUnit.SECONDS)
.readTimeout(60, TimeUnit.SECONDS)
.writeTimeout(60, TimeUnit.SECONDS)
.build();

其中callTimeout,readTimeout,writeTimeout和okio的AsyncTimeout有著密不可分的關(guān)系,其內(nèi)部維護(hù)了一個(gè)Watchdog,單獨(dú)開一個(gè)線程死循環(huán)判斷是否超時(shí)


image.png

connectTimeout:

指的是建立連接所用的時(shí)間震叙,適用于網(wǎng)絡(luò)狀況正常的情況下松嘶,兩端連接所用的時(shí)間忆畅。
通過跟源碼發(fā)現(xiàn)這個(gè)值用在了 socket.connect(address, connectTimeout);

callTimeout:

這個(gè)值從調(diào)用call.execute();和enqueue();這兩個(gè)方法開始計(jì)時(shí),時(shí)間到后網(wǎng)絡(luò)還未請(qǐng)求完成將調(diào)用cancel();方法
在RealCall類中可以看到在構(gòu)造方法中創(chuàng)建timeout匿名內(nèi)部類


創(chuàng)建.png

在execute方法中開始計(jì)時(shí)


調(diào)用.png

在timeoutExit方法中結(jié)束計(jì)時(shí)
image.png

pingInterval

通過跟源碼我們可以看到,這個(gè)值只有http2和webSocket中有使用


image.png

image.png

如果設(shè)置了這個(gè)值會(huì)定時(shí)的向服務(wù)器發(fā)送一個(gè)消息來保持長連接


image.png

image.png

image.png

所以在寫websocket時(shí)是完全可以只用設(shè)置這個(gè)值來保持長連接的.
客戶端在發(fā)送ping消息時(shí)服務(wù)端會(huì)相應(yīng)的返回pong消息來進(jìn)行回應(yīng).同時(shí)okhttp也實(shí)現(xiàn)了pong,服務(wù)端在發(fā)起ping的時(shí)候客戶端會(huì)通過pong來進(jìn)行回應(yīng),即:在進(jìn)行長連接時(shí),客戶端不需要進(jìn)行只需要服務(wù)端進(jìn)行定時(shí)ping也是可以保持長連接的.

接下來就開始講和我們密切相關(guān)的readTimeout和writeTimeout了,當(dāng)然也是最復(fù)雜的.其中最重要的還是readTimeout,我們先看writeTimeout

writeTimeout

這個(gè)值大致有3個(gè)地方用到


image.png

其中第二處和第三處的用用法是一致的,最后都是調(diào)用了
sink.timeout().timeout(writeTimeout, MILLISECONDS);
這寫到底是什么意思呢?
這個(gè)就不得不說okio了,okhttp中幾乎所有的流的操作都是由okio完成的,在okio.AsyncTimeout中對(duì)Sink(類似于OutputStream)和Source(類似于InputStream)進(jìn)行了一層封裝


image.png
image.png

/**

  • Don't write more than 64 KiB of data at a time, give or take a segment. Otherwise slow
  • connections may suffer timeouts even when they're making (slow) progress. Without this, writing
  • a single 1 MiB buffer may never succeed on a sufficiently slow connection.
    */
    private static final int TIMEOUT_WRITE_SIZE = 64 * 1024;

這其中的邏輯還是相當(dāng)復(fù)雜的,大致意思就是所有的sink都被封裝了一個(gè)超時(shí)機(jī)制,需要在我們?cè)O(shè)置的時(shí)間內(nèi)寫出TIMEOUT_WRITE_SIZE(64k)的數(shù)據(jù),如果無法完成即為超時(shí),所以,我們?cè)谏洗挝募r(shí)明明只設(shè)置了幾十秒的超時(shí)時(shí)間卻不會(huì)超時(shí).

在http2中就沒有再使用okio的超時(shí)機(jī)制了,當(dāng)然超時(shí)計(jì)時(shí)器還是用的AsyncTimeout.的Watchdog

image.png
image.png

image.png
image.png

image.png
image.png

image.png
image.png

可以看到,在http2中采用的是線程等待的策略

readTimeout

readTimeout和writeTimeout幾乎完全一樣,只是操作相反,而且header的讀取和body的讀取是分開進(jìn)行的,由于header數(shù)據(jù)量較小就不用討論了.
okio中每次讀取不大于8k.
final class Segment {
/** The size of all segments in bytes. */
static final int SIZE = 8192;


image.png
image.png

http2中每次讀取不大于8k.


image.png
image.png

然后還漏了一點(diǎn):


image.png

socket.setSoTimeout(chain.readTimeoutMillis());
這行代碼什么意思呢?
setSotimeout(10000)是表示如果對(duì)方連接狀態(tài)10秒沒有收到數(shù)據(jù)的話強(qiáng)制斷開客戶端。
如果想要長連接的話晚顷,可以使用心跳包來通知服務(wù)器,也就是我沒有發(fā)給你數(shù)據(jù)疗疟,但是我告訴你我還活著.

最后,如果超時(shí)時(shí)間設(shè)置的如果是0,那么代表超時(shí)時(shí)長為無限.

附上okhttp的默認(rèn)超時(shí)時(shí)間


image.png
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末该默,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子策彤,更是在濱河造成了極大的恐慌栓袖,老刑警劉巖顿膨,帶你破解...
    沈念sama閱讀 211,123評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異叽赊,居然都是意外死亡恋沃,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門必指,熙熙樓的掌柜王于貴愁眉苦臉地迎上來囊咏,“玉大人,你說我怎么就攤上這事塔橡∶犯睿” “怎么了?”我有些...
    開封第一講書人閱讀 156,723評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵葛家,是天一觀的道長户辞。 經(jīng)常有香客問我,道長癞谒,這世上最難降的妖魔是什么底燎? 我笑而不...
    開封第一講書人閱讀 56,357評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮弹砚,結(jié)果婚禮上双仍,老公的妹妹穿的比我還像新娘。我一直安慰自己桌吃,他們只是感情好朱沃,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,412評(píng)論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著茅诱,像睡著了一般逗物。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上瑟俭,一...
    開封第一講書人閱讀 49,760評(píng)論 1 289
  • 那天翎卓,我揣著相機(jī)與錄音,去河邊找鬼尔当。 笑死莲祸,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的椭迎。 我是一名探鬼主播锐帜,決...
    沈念sama閱讀 38,904評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼畜号!你這毒婦竟也來了缴阎?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,672評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤简软,失蹤者是張志新(化名)和其女友劉穎蛮拔,沒想到半個(gè)月后述暂,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,118評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡建炫,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,456評(píng)論 2 325
  • 正文 我和宋清朗相戀三年畦韭,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片肛跌。...
    茶點(diǎn)故事閱讀 38,599評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡艺配,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出衍慎,到底是詐尸還是另有隱情转唉,我是刑警寧澤,帶...
    沈念sama閱讀 34,264評(píng)論 4 328
  • 正文 年R本政府宣布稳捆,位于F島的核電站赠法,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏乔夯。R本人自食惡果不足惜砖织,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,857評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望驯嘱。 院中可真熱鬧镶苞,春花似錦喳坠、人聲如沸鞠评。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽剃幌。三九已至,卻和暖如春晾浴,著一層夾襖步出監(jiān)牢的瞬間负乡,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評(píng)論 1 264
  • 我被黑心中介騙來泰國打工脊凰, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留抖棘,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,286評(píng)論 2 360
  • 正文 我出身青樓狸涌,卻偏偏與公主長得像切省,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子帕胆,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,465評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容