一步步帶你讀懂 Okhttp 源碼
- 2019 年 10 月 25 日
- 筆記
版權聲明:本文為部落客原創文章,遵循 CC 4.0 BY-SA 版權協議,轉載請附上原文出處鏈接和本聲明。
本文鏈接:https://blog.csdn.net/gdutxiaoxu/article/details/100545600
前言

okHttp, square 公司開源的網路請求神器,截止到 2019-09-02,在 Github 上面已經超過 34K 的 star,足見他的受歡迎程度。
到目前為止,他的最新版本是 4.1.0, 使用 kotlin 語言寫的,由於本人對 kotlin 語言不是很熟悉,這篇文章已 3.5.0 的版本為基礎進行分析。
簡介
Rxjava+Okhttp+Refrofit 如今已經成為項目網路請求的首選,在講解原理之前,我們先來看一下 Okhttp 的基本使用。
使用 OkHttp 基本是以下四步:
- 創建 OkHttpClient 對象
- 創建Request請求對象
- 創建Call對象
- 同步請求調用call.execute();非同步請求調用call.enqueue(callback)
同步執行
//創建OkHttpClient對象 OkHttpClient client = new OkHttpClient(); String run(String url) throws IOException { //創建Request請求對象 Request request = new Request.Builder() .url(url) .build(); //創建Call對象,並執行同步獲取網路數據 Response response = client.newCall(request).execute(); return response.body().string(); }
非同步執行
void runAsync(String url, Callback callback) { OkHttpClient client = new OkHttpClient.Builder().addInterceptor(new Interceptor() { @Override public Response intercept(Chain chain) throws IOException { Request request = chain.request(); Request.Builder builder = request.newBuilder().addHeader("name", "test"); return chain.proceed(builder.build()); } }).build(); //創建Request請求對象 Request request = new Request.Builder() .url(url) .build(); client.newCall(request).enqueue(callback); }
接下來我會從這四步,分析 Okhttp 的基本原理。
OkHttpClient
創建 OkHttpClient 一般有兩種方法,一種是直接 new OkHttpClient()
,另外一種是通過 OkHttpClient.Builder()
OkhttpClient client = new OkHttpClient .Builder() .connectTimeout(5, TimeUnit.SECONDS) .writeTimeout(10,TimeUnit.SECONDS) .readTimeout(10, TimeUnit.SECONDS) .build();
第二種創建方式主要是通過建造者模式,來配置一些參數,比如連接超時時間,讀寫超時時間,超時重試次數等。這樣有一個好處,可以對外屏蔽掉構建 client 的細節。
Request
public final class Request { final HttpUrl url; final String method; final Headers headers; final RequestBody body; final Object tag; private volatile CacheControl cacheControl; // Lazily initialized. }
Request 對象主要封裝的是一些網路請求的資訊,比如請求 url,請求方法,請求頭,請求 body 等,也比較簡單,這裡不再展開闡述。
Call 對象
@Override public Call newCall(Request request) { return new RealCall(this, request, false /* for web socket */); }
可以看到 call 對象實際是 RealCall 的實例化對象
RealCall#execute()
@Override public Response execute() throws IOException { synchronized (this) { if (executed) throw new IllegalStateException("Already Executed"); executed = true; } captureCallStackTrace(); try { // 執行 client.dispatcher() 的 executed 方法 client.dispatcher().executed(this); Response result = getResponseWithInterceptorChain(); if (result == null) throw new IOException("Canceled"); return result; } finally { // 最後再執行 dispatcher 的 finish 方法 client.dispatcher().finished(this); } }
在 execute 方法中,首先會調用 client.dispatcher().executed(this)
加入到 runningAsyncCalls 隊列當中,接著執行 getResponseWithInterceptorChain()
獲取請求結果,最終再執行 client.dispatcher().finished(this)
將 realCall 從 runningAsyncCalls 隊列中移除 。
我們先來看一下 getResponseWithInterceptorChain
方法
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); return chain.proceed(originalRequest); }
可以看到,首先,他會將客戶端的 interceptors 添加到 list 當中,接著,再添加 okhhttp 裡面的 interceptor,然後構建了一個 RealInterceptorChain
對象,並將我們的 List<Interceptor>
作為成員變數,最後調用 RealInterceptorChain 的 proced 方法。
- client.interceptors() -> 我們自己添加的請求攔截器,通常是做一些添加統一的token之類操作
- retryAndFollowUpInterceptor -> 主要負責錯誤重試和請求重定向
- BridgeInterceptor -> 負責添加網路請求相關的必要的一些請求頭,比如Content-Type、Content-Length、Transfer-Encoding、User-Agent等等
- CacheInterceptor -> 負責處理快取相關操作
- ConnectInterceptor -> 負責與伺服器進行連接的操作
- networkInterceptors -> 同樣是我們可以添加的攔截器的一種,它與client.interceptors() 不同的是二者攔截的位置不一樣。
- CallServerInterceptor -> 在這個攔截器中才會進行真實的網路請求
Interceptor 裡面是怎樣實現的,這裡我們暫不討論,接下來,我們來看一下 proceed 方法
proceed 方法
public Response proceed(Request request, StreamAllocation streamAllocation, HttpCodec httpCodec, Connection connection) throws IOException { // 省略無關程式碼 // 生成 list 當中下一個 interceptot 的 chain 對象 RealInterceptorChain next = new RealInterceptorChain( interceptors, streamAllocation, httpCodec, connection, index + 1, request); // 當前的 interceptor Interceptor interceptor = interceptors.get(index); // 當前的 intercept 處理下一個 intercept 包裝的 chain 對象 Response response = interceptor.intercept(next); // ---- return response; }
proceed 方法也很簡單,proceed方法每次從攔截器列表中取出攔截器,並調用 interceptor.intercept(next)。
熟悉 Okhttp 的應該都在回到,我們在 addInterceptor 創建 Interceptor 實例,最終都會調用 chain.proceed(Request request)
,從而形成一種鏈式調用。關於責任鏈模式的可以看我的這一篇文章 責任鏈模式以及在 Android 中的應用
OkHttpClient client = new OkHttpClient.Builder().addInterceptor(new Interceptor() { @Override public Response intercept(Chain chain) throws IOException { Request request = chain.request(); Request.Builder builder = request.newBuilder().addHeader("name","test"); return chain.proceed(builder.build()); } }).build();

而 OkHttp 是怎樣結束循環調用的,這是因為最後一個攔截器 CallServerInterceptor 並沒有調用chain.proceed(request),所以能夠結束循環調用。
dispatcher
public final class Dispatcher { private int maxRequests = 64; private int maxRequestsPerHost = 5; private Runnable idleCallback; /** Executes calls. Created lazily. */ private ExecutorService executorService; // 非同步的請求等待隊列 private final Deque<AsyncCall> readyAsyncCalls = new ArrayDeque<>(); // 非同步的正在請求的隊列 private final Deque<AsyncCall> runningAsyncCalls = new ArrayDeque<>(); // 絨布的正在請求的隊列 private final Deque<RealCall> runningSyncCalls = new ArrayDeque<>(); }
非同步請求 enqueue(Callback responseCallback)
@Override public void enqueue(Callback responseCallback) { synchronized (this) { if (executed) throw new IllegalStateException("Already Executed"); executed = true; } captureCallStackTrace(); client.dispatcher().enqueue(new AsyncCall(responseCallback)); }
首先,我們先來看一下 AsyncCall 這個類
final class AsyncCall extends NamedRunnable { private final Callback responseCallback; // ---- @Override protected void execute() { boolean signalledCallback = false; try { Response response = getResponseWithInterceptorChain(); // 判斷請求是否取消了,如果取消了,直接回調 onFailure if (retryAndFollowUpInterceptor.isCanceled()) { signalledCallback = true; responseCallback.onFailure(RealCall.this, new IOException("Canceled")); } else { // 請求成功 signalledCallback = true; responseCallback.onResponse(RealCall.this, response); } } catch (IOException e) { if (signalledCallback) { // Do not signal the callback twice! Platform.get().log(INFO, "Callback failure for " + toLoggableString(), e); } else { responseCallback.onFailure(RealCall.this, e); } } finally { client.dispatcher().finished(this); } } }
public abstract class NamedRunnable implements Runnable { protected final String name; public NamedRunnable(String format, Object... args) { this.name = Util.format(format, args); } @Override public final void run() { String oldName = Thread.currentThread().getName(); Thread.currentThread().setName(name); try { execute(); } finally { Thread.currentThread().setName(oldName); } } protected abstract void execute(); }
可以看到 AsyncCall 繼承 NamedRunnable, 而 NamedRunnable 是 Runnable 的子類,當執行 run 方法時,會執行 execute 方法。
我們再來看一下 dispatcher 的 enqueue 方法
synchronized void enqueue(AsyncCall call) { if (runningAsyncCalls.size() < maxRequests && runningCallsForHost(call) < maxRequestsPerHost) { runningAsyncCalls.add(call); executorService().execute(call); } else { readyAsyncCalls.add(call); } }
可以看到當正在請求的請求數小於 maxRequests
的時候並且當前正在請求的隊列裡面相當 host 的小於 maxRequestsPerHost
, 直接添加到 runningAsyncCalls
隊列中,並添加到執行緒池裡面執行,否則添加到準備隊列 readyAsyncCalls
裡面。
當執行 executorService().execute(call)
的時候,會調用 run 方法, run 方法又會調用到 execute 方法進行網路請求,請求完成之後,會調用 client.dispatcher().finished(this)
從隊列裡面移除。
到此, Okhttp 的主要流程已經講完
小結
- 有一個分發器 Dispatcher,裡面有三個請求隊列,一個是正在請求的隊列,一個是等待隊列,另外一個是同步的正在請求的隊列,當我們執行 enqueue 方法的時候,他會判斷正在請求隊列數量是否超過允許的最大並發數量(默認是 64)(執行緒池的原理),如果超過了,會添加到等待隊列裡面。 excute 方法是同步執行的,每次執行會添加到同步請求隊列當中,執行完畢之後會移除
- 設計的核心思想責任鏈模式,當我們需要攔截的時候,可以實現 Interceptor 介面,會按照添加的順序執行 Chain.proceed 方法。
- 職責分明,OkhttpClient 對象主要處理一些基礎的配置,比如連接超時,讀寫超時,添加攔截器。Request 主要配置請求方法,請求頭等。