OKHttp源碼學(xué)習(xí)筆記:調(diào)用流程

基本用法

String url = "http://wwww.baidu.com";
OkHttpClient okHttpClient = new OkHttpClient();
final Request request = new Request.Builder()
        .url(url)
        .post()
        .build();
Call call = okHttpClient.newCall(request);

//異步請求
call.enqueue(new Callback() {
    @Override
    public void onFailure(Call call, IOException e) {
        Log.d(TAG, "onFailure: 異步請求失敗 ");
    }

    @Override
    public void onResponse(Call call, Response response) throws IOException {
        Log.d(TAG, "onResponse: 異步請求成功 " + response.body().string());
    }
});

//同步請求
Response  response = client.newCall(request).execute();//得到Response 對象(會阻塞當(dāng)前線程)

跟到深處會發(fā)現(xiàn)無論同步和異步請求的關(guān)鍵都是攔截器方法

 Response getResponseWithInterceptorChain() throws IOException {
    // Build a full stack of interceptors.
    List<Interceptor> interceptors = new ArrayList<>();
    interceptors.addAll(client.interceptors());
    interceptors.add(retryAndFollowUpInterceptor);
    interceptors.add(new BridgeInterceptor(client.cookieJar()));
    interceptors.add(new CacheInterceptor(client.internalCache()));
    interceptors.add(new ConnectInterceptor(client));
    if (!forWebSocket) {
      interceptors.addAll(client.networkInterceptors());
    }
    interceptors.add(new CallServerInterceptor(forWebSocket));

    Interceptor.Chain chain = new RealInterceptorChain(interceptors, null, null, null, 0,
        originalRequest, this, eventListener, client.connectTimeoutMillis(),
        client.readTimeoutMillis(), client.writeTimeoutMillis());

    return chain.proceed(originalRequest);
  }

接下來就以該方法為入口分析下OkHttp3的代碼

攔截器調(diào)用流程

再貼貼上面的代碼分析一下

 //RealCall.Java
 Response getResponseWithInterceptorChain() throws IOException {
    // Build a full stack of interceptors.
    // 建立攔截器列表
    List<Interceptor> interceptors = new ArrayList<>();
    interceptors.addAll(client.interceptors());//自定義的攔截器,例如日志打印攔截器
    interceptors.add(retryAndFollowUpInterceptor);//重連及重定向攔截器
    interceptors.add(new BridgeInterceptor(client.cookieJar()));//橋攔截器?主要是用來加Header信息
    interceptors.add(new CacheInterceptor(client.internalCache()));//緩存攔截器
    interceptors.add(new ConnectInterceptor(client));//連接攔截器,用于建立連接
    if (!forWebSocket) {
      interceptors.addAll(client.networkInterceptors());//僅在建立WebSocket時才會加的攔截器
    }
    interceptors.add(new CallServerInterceptor(forWebSocket));//發(fā)送信息攔截器(之前的連接攔截器負(fù)責(zé)建立連接但是啥也沒干,這個攔截器負(fù)責(zé)給服務(wù)器發(fā)消息并接收返回)

    Interceptor.Chain chain = new RealInterceptorChain(interceptors, null, null, null, 0,
        originalRequest, this, eventListener, client.connectTimeoutMillis(),
        client.readTimeoutMillis(), client.writeTimeoutMillis());
   
    return chain.proceed(originalRequest);//開啟鏈?zhǔn)秸{(diào)用
  }

這里主要是攔截器的大概調(diào)用流程,首先添加自定義的攔截器(一般是日志打印攔截器或者再增加Header信息之類的),然后添加其他固定的攔截器,注意這里的攔截器順序都是固定的,添加后用一個RealInterceptorChain包裝并依次調(diào)用,執(zhí)行順序就是添加攔截器的順序,這里每個的攔截器都是繼承了Interceptor接口

public interface Interceptor {
  //攔截器子類 需復(fù)寫的方法,基本每個攔截器的具體用法都在這個函數(shù)里面
  Response intercept(Chain chain) throws IOException;

  //攔截器調(diào)用鏈繼承接口
  interface Chain {
    Request request();

    Response proceed(Request request) throws IOException;

    @Nullable Connection connection();

    Call call();

    int connectTimeoutMillis();

    Chain withConnectTimeout(int timeout, TimeUnit unit);

    int readTimeoutMillis();

    Chain withReadTimeout(int timeout, TimeUnit unit);

    int writeTimeoutMillis();

    Chain withWriteTimeout(int timeout, TimeUnit unit);
  }
}

再看看Chain子類RealInterceptorChain的關(guān)鍵方法proceed

  //帶Response返回,方便做二次處理
  @Override public Response proceed(Request request) throws IOException {
    return proceed(request, streamAllocation, httpCodec, connection);
  
  }

  //這里僅貼出關(guān)鍵代碼,該函數(shù)其他代碼都是條件判斷和返回處理
  public Response proceed(Request request, StreamAllocation streamAllocation, HttpCodec httpCodec,
      RealConnection connection) throws IOException {
    
   .........

    // 調(diào)用核心:直接index+1其他參數(shù)不變來調(diào)用下一個Interceptor
    RealInterceptorChain next = new RealInterceptorChain(interceptors, streamAllocation, httpCodec,
        connection, index + 1, request, call, eventListener, connectTimeout, readTimeout,
        writeTimeout);
    Interceptor interceptor = interceptors.get(index);
    Response response = interceptor.intercept(next);
   
    ...........

    return response;
  }

由此看出該方法主要是負(fù)責(zé)調(diào)用鏈的運(yùn)轉(zhuǎn),當(dāng)某個Interceptor在處理好大部分工作(注意這里說大部分是因?yàn)橥ㄟ^后面的Interceptor返回可以再做第二次處理,類似與Android View的觸摸事件下發(fā))后直接把原來的request丟過去調(diào)用chain.proceed(request),RealInterceptorChain類就會自動通過index+1來調(diào)用下一個Interceptor
源葫。這里也可以看出雖然方法內(nèi)一直next next next的 又有Chain這樣的字眼讓人誤以為內(nèi)部是鏈表結(jié)構(gòu)存放Interceptor,其實(shí)是通過列表存放的
這里順帶科普一下列表和鏈表的大概優(yōu)缺點(diǎn)(給自己看。胁镐。。诸衔。)
https://blog.csdn.net/st1441517927/article/details/99483738

總結(jié)

OkHttp通過攔截器設(shè)計模式非常的巧妙,一來是細(xì)化各個分功能解耦度高,層次分明,二來是方便定制(可以自定義攔截器), 再者采用類似Android 觸摸事件分發(fā)的機(jī)制,使得每個Interceptor可以進(jìn)行二次處理,非常關(guān)鍵,某些需要二次處理(或重點(diǎn)處理返回那一次)的就得益于這種機(jī)制,例如重定向攔截器和自定義的日志攔截器尤為明顯

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末盯漂,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子笨农,更是在濱河造成了極大的恐慌就缆,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,948評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件谒亦,死亡現(xiàn)場離奇詭異竭宰,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)份招,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,371評論 3 385
  • 文/潘曉璐 我一進(jìn)店門切揭,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人锁摔,你說我怎么就攤上這事廓旬。” “怎么了谐腰?”我有些...
    開封第一講書人閱讀 157,490評論 0 348
  • 文/不壞的土叔 我叫張陵孕豹,是天一觀的道長。 經(jīng)常有香客問我十气,道長励背,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,521評論 1 284
  • 正文 為了忘掉前任砸西,我火速辦了婚禮叶眉,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘芹枷。我一直安慰自己衅疙,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,627評論 6 386
  • 文/花漫 我一把揭開白布杖狼。 她就那樣靜靜地躺著,像睡著了一般妖爷。 火紅的嫁衣襯著肌膚如雪蝶涩。 梳的紋絲不亂的頭發(fā)上理朋,一...
    開封第一講書人閱讀 49,842評論 1 290
  • 那天,我揣著相機(jī)與錄音绿聘,去河邊找鬼嗽上。 笑死,一個胖子當(dāng)著我的面吹牛熄攘,可吹牛的內(nèi)容都是我干的兽愤。 我是一名探鬼主播,決...
    沈念sama閱讀 38,997評論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼挪圾,長吁一口氣:“原來是場噩夢啊……” “哼浅萧!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起哲思,我...
    開封第一講書人閱讀 37,741評論 0 268
  • 序言:老撾萬榮一對情侶失蹤洼畅,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后棚赔,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體帝簇,經(jīng)...
    沈念sama閱讀 44,203評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,534評論 2 327
  • 正文 我和宋清朗相戀三年靠益,在試婚紗的時候發(fā)現(xiàn)自己被綠了丧肴。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,673評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡胧后,死狀恐怖芋浮,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情绩卤,我是刑警寧澤途样,帶...
    沈念sama閱讀 34,339評論 4 330
  • 正文 年R本政府宣布,位于F島的核電站濒憋,受9級特大地震影響何暇,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜凛驮,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,955評論 3 313
  • 文/蒙蒙 一裆站、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧黔夭,春花似錦宏胯、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,770評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至婚惫,卻和暖如春氛赐,著一層夾襖步出監(jiān)牢的瞬間魂爪,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,000評論 1 266
  • 我被黑心中介騙來泰國打工艰管, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留滓侍,地道東北人。 一個月前我還...
    沈念sama閱讀 46,394評論 2 360
  • 正文 我出身青樓牲芋,卻偏偏與公主長得像撩笆,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子缸浦,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,562評論 2 349

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