源碼分析筆記——OkHttp

1.寫在前面

工做以來,大部分時間都在忙於各類業務。業務的實現當然重要,不過我仍然堅信注重基礎知識的積累纔是正道。想一想用了這麼久的OkHttp居然沒有仔細品讀他的源碼,深感慚愧。是時候研究一下了。java

2.關於OkHttp

2.1 簡介

引用項目官方的一句話:An HTTP & HTTP/2 client for Android and Java applications。OkHttp是一個處理網絡請求的高性能框架,由Square公司貢獻(該公司還貢獻了Picasso) 項目首頁地址git

2.2 優點

  • 共享Socket,減小對服務器的請求次數
  • 經過鏈接池,減小了請求延遲;
  • 支持GZIP來減小數據流量;
  • 緩存響應數據來減小徹底重複的網絡請求;
  • 使用 OkHttp 無需重寫您程序中的網絡代碼。OkHttp實現了幾乎和java.net.HttpURLConnection同樣的API;
  • OkHttp會從不少經常使用的鏈接問題中自動恢復。若是您的服務器配置了多個IP地址,當第一個IP鏈接失敗的時候,會自動嘗試下一個IP。

3.開始啃源碼

注:本文代碼梳理基於 v3.10.0 版本,建議從github上clone一份代碼而後跟着本文一塊兒梳理一遍。github

3.1 先描述下大致的分析步驟

本文會以請求流程爲主線梳理OKHttp的源碼,之因此不深刻代碼細節是由於過度深鑽代碼實現自己沒有太多的指導意義,而且容易讓初學者陷入只見樹木不見森林的尷尬場面。因爲OkHttp發起請求的時候從是否同步的角度來說能夠分爲同步或異步。全部接下來的源碼梳理也從這兩個方面來分析。先上一張流程圖:web

上圖來自 Piasy的博客。這張圖對OkHttp的流程的詳細程度讓我實在找不出重畫的理由,因此照搬過來啦(實際上是我懶)。

3.2 同步請求

3.2.1 如何發起同步請求?

完整代碼以下:緩存

OkHttpClient client = new OkHttpClient
                        .Builder().build();

String run(String url) throws IOException {
  Request request = new Request.Builder()
      .url(url)
      .build();

  Response response = client.newCall(request).execute();
  return response.body().string();
}
複製代碼

3.2.2 同步請求涉及到的類

按照3.2.1小節的demo,咱們先來分析用到的這幾個類分別扮演了什麼樣的角色,而後按照請求的流程將全部知識串聯起來。bash

3.2.2.1 Request類

引用一下官方的類描述: An HTTP request. Instances of this class are immutable if their is null or itself immutable.很明顯,這個類表明一個HTTP請求。來看下他的源碼:服務器

public final class Request {
  final HttpUrl url;
  final String method;
  final Headers headers;
  final @Nullable RequestBody body;
  final Object tag;

  private volatile CacheControl cacheControl; // Lazily initialized.

  Request(Builder builder) {
    this.url = builder.url;
    this.method = builder.method;
    this.headers = builder.headers.build();
    this.body = builder.body;
    this.tag = builder.tag != null ? builder.tag : this;
  }
  ...
  public static class Builder {
    HttpUrl url;
    String method;
    Headers.Builder headers;
    RequestBody body;
    Object tag;

    public Builder() {
      this.method = "GET";
      this.headers = new Headers.Builder();
    }

    Builder(Request request) {
      this.url = request.url;
      this.method = request.method;
      this.body = request.body;
      this.tag = request.tag;
      this.headers = request.headers.newBuilder();
    }

    public Builder url(HttpUrl url) {
      if (url == null) throw new NullPointerException("url == null");
      this.url = url;
      return this;
    }
    ...
複製代碼

上述代碼通過篩選,我們只看關鍵的部分。由代碼可見,Request的主要做用是用來描述一次HTTP請求,這個類中包含請求的地址、請求方法、請求體等類型數據,沒什麼難度。強調一點:構建Request這個類的使用須要使用內部類Builder,很明顯,典型的建造器模式。cookie

3.2.2.2 OkHttpClient類

顧名思義,這個類是OKHttp的一個「客戶端角色」,按照官方介紹,他實際上是一個「Call」的工廠。什麼是「Call」呢?實際上是一次請求的過程,這裏先略過,後面再介紹。一塊兒來看下OkHttpClient的源碼。網絡

public class OkHttpClient implements Cloneable, Call.Factory, WebSocket.Factory {

   public OkHttpClient() {
    this(new Builder());
  }

  OkHttpClient(Builder builder) {
    this.dispatcher = builder.dispatcher;
    ...
  }
  
 ...

   public static final class Builder {
    Dispatcher dispatcher;
    @Nullable Proxy proxy;
    List<Protocol> protocols;
    List<ConnectionSpec> connectionSpecs;
    final List<Interceptor> interceptors = new ArrayList<>();
    final List<Interceptor> networkInterceptors = new ArrayList<>();
    EventListener.Factory eventListenerFactory;
    ProxySelector proxySelector;
    CookieJar cookieJar;
    @Nullable Cache cache;
    @Nullable InternalCache internalCache;
    SocketFactory socketFactory;
    @Nullable SSLSocketFactory sslSocketFactory;
    @Nullable CertificateChainCleaner certificateChainCleaner;
    HostnameVerifier hostnameVerifier;
    CertificatePinner certificatePinner;
    Authenticator proxyAuthenticator;
    Authenticator authenticator;
    ConnectionPool connectionPool;
    Dns dns;
    boolean followSslRedirects;
    boolean followRedirects;
    boolean retryOnConnectionFailure;
    int connectTimeout;
    int readTimeout;
    int writeTimeout;
    int pingInterval;

    public Builder() {
      dispatcher = new Dispatcher();
     ...
    }
}
複製代碼

上述代碼通過精簡,一樣的,我們只挑重點的看,OkHttpClient一樣提供了建造器模式來建立一個實例,主要對攔截器、讀寫超時等屬性進行配置。這裏要注意,OkHttpClient的Builder已經爲咱們提供了默認的配置,咱們只須要按需重寫自定義的屬性就好。app

3.2.2.3 Call類

直白來說Call表明一次請求的過程,按照官方描述,這個類表明一個已經準備好將要被執行的請求。而且同一個Call實例不能夠被執行兩次。這裏注意一點,Call能夠經過原型模式複製一個與本身相同內容的對象。

/**
 * A call is a request that has been prepared for execution. A call can be canceled. As this object
 * represents a single request/response pair (stream), it cannot be executed twice.
 */
public interface Call extends Cloneable {
  Request request();

  Response execute() throws IOException;

  void cancel();

  boolean isExecuted();

  boolean isCanceled();

  Call clone();

  interface Factory {
    Call newCall(Request request);
  }
}
複製代碼

3.2.3 同步請求流程

按照3.2.1小節的demo,一塊兒來梳理一下同步請求的流程,順便把3.2.2小節中提到的知識點串聯一下。 發起請求的時候,典型應用代碼爲:Response response = client.newCall(request).execute(); 因此一塊兒來看下OkHttpClientnewCall方法作了什麼事情:

OkHttpClient的newCall方法內容:

/**
   * Prepares the {@code request} to be executed at some point in the future.
   */
  @Override public Call newCall(Request request) {
    return RealCall.newRealCall(this, request, false /* for web socket */);
  }

複製代碼

上述代碼能夠看出newCall方法直接調用了RealCall.newRealCall方法。這個RealCall類實現了Call這個接口。來看下RealCall.newRealCall的內容。

RealCall.newRealCall方法體:

static RealCall newRealCall(OkHttpClient client, Request originalRequest, boolean forWebSocket) {
    // Safely publish the Call instance to the EventListener.
    RealCall call = new RealCall(client, originalRequest, forWebSocket);
    call.eventListener = client.eventListenerFactory().create(call);
    return call;
  }
複製代碼

上述代碼很清晰,newRealCall方法直接new了一個RealCall對象而且返回了。

分析到這裏,我們接着最開始的典型應用代碼分析Response response = client.newCall(request).execute();綜合上述的代碼分析,這段代碼能夠替換成僞代碼Response response = RealCall.execute();。因此下面一塊兒來看下RealCall中的execute方法吧。

RealCall.execute方法:

@Override public Response execute() throws IOException {
    synchronized (this) {
      if (executed) throw new IllegalStateException("Already Executed");
      executed = true;
    }
    captureCallStackTrace();
    eventListener.callStart(this);
    try {
      client.dispatcher().executed(this);
      Response result = getResponseWithInterceptorChain();
      if (result == null) throw new IOException("Canceled");
      return result;
    } catch (IOException e) {
      eventListener.callFailed(this, e);
      throw e;
    } finally {
      client.dispatcher().finished(this);
    }
複製代碼

重點分析上述代碼兩個地方:

  • client.dispatcher().executed(this);,這句代碼中用client中的Dispatcher對當前Call進行了一次標記。這個方法內部用一個名叫runningSyncCalls的Deque對當前的Call進行了存儲。因爲Dispatcher在同步調用的過程當中並非主角,因此這裏先佔個坑後面分析異步調用的時候在分析一下。
  • Response result = getResponseWithInterceptorChain();我我的認爲,這句調用纔是同步調用乃至整個OkHttp的靈魂所在。下面重點分析一下。

來看下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, this, eventListener, client.connectTimeoutMillis(),
        client.readTimeoutMillis(), client.writeTimeoutMillis());

    return chain.proceed(originalRequest);
  }
複製代碼

想必你們即便沒有閱讀過OkHttp的源碼,必定也有所耳聞,OkHttp的精髓在於攔截器的使用。沒錯getResponseWithInterceptorChain這個方法就是負責將責任鏈上的每個攔截器進行串聯的方法。該方法的邏輯也比較通俗易懂,將開發者定義的攔截器與OkHttp內置的攔截器封裝成一個List而後塞給了RealInterceptorChain,最後執行RealInterceptorChain的proceed方法。那這個RealInterceptorChain是什麼呢?通俗一點來說,這個類是一個具體的攔截器鏈,沒錯這是典型的責任鏈模式。

RealInterceptorChain類代碼:

/**
 * A concrete interceptor chain that carries the entire interceptor chain: all application
 * interceptors, the OkHttp core, all network interceptors, and finally the network caller.
 */
public final class RealInterceptorChain implements Interceptor.Chain {
  private final List<Interceptor> interceptors;
  private final StreamAllocation streamAllocation;
  private final HttpCodec httpCodec;
  private final RealConnection connection;
  private final int index;
  private final Request request;
  private final Call call;
  private final EventListener eventListener;
  private final int connectTimeout;
  private final int readTimeout;
  private final int writeTimeout;
  private int calls;

  public RealInterceptorChain(List<Interceptor> interceptors, StreamAllocation streamAllocation,
      HttpCodec httpCodec, RealConnection connection, int index, Request request, Call call,
      EventListener eventListener, int connectTimeout, int readTimeout, int writeTimeout) {
    this.interceptors = interceptors;
    this.connection = connection;
    ...
  }


  public Response proceed(Request request, StreamAllocation streamAllocation, HttpCodec httpCodec,
      RealConnection connection) throws IOException {
    if (index >= interceptors.size()) throw new AssertionError();

    calls++;

    // If we already have a stream, confirm that the incoming request will use it.
    if (this.httpCodec != null && !this.connection.supportsUrl(request.url())) {
      throw new IllegalStateException("network interceptor " + interceptors.get(index - 1)
          + " must retain the same host and port");
    }

    // If we already have a stream, confirm that this is the only call to chain.proceed().
    if (this.httpCodec != null && calls > 1) {
      throw new IllegalStateException("network interceptor " + interceptors.get(index - 1)
          + " must call proceed() exactly once");
    }

    // Call the next interceptor in the chain.
    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);

    // Confirm that the next interceptor made its required call to chain.proceed().
    if (httpCodec != null && index + 1 < interceptors.size() && next.calls != 1) {
      throw new IllegalStateException("network interceptor " + interceptor
          + " must call proceed() exactly once");
    }

    // Confirm that the intercepted response isn't null. if (response == null) { throw new NullPointerException("interceptor " + interceptor + " returned null"); } if (response.body() == null) { throw new IllegalStateException( "interceptor " + interceptor + " returned a response with no body"); } return response; } } 複製代碼

一開始分析到這裏的時候,我有點蒙圈。弄了很久沒弄懂這個責任鏈是怎麼執行的。最後仍是搬出了斷點打法,終於看到了光明。再複雜的責任鏈(因爲本人比較水,因此以爲OkHttp的責任鏈是比較複雜的,歡迎拍磚)把握好兩點就夠了:

  • 怎樣獲取的責任鏈上下一環的實例
  • 責任鏈是怎樣串行執行的

帶着上述的問題進行斷點大法,終於找到了如下核心代碼:

// Call the next interceptor in the chain.
    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);
複製代碼

這段代碼使用遞歸調用,若是責任鏈上還有下一環的攔截器實例,則取出該實例,而且調用該實例的intercept方法。前面分析getResponseWithInterceptorChain這個方法的時候對攔截器進行過介紹。OkHttp中全部攔截器都須要實現Interceptor這個接口。 這個接口惟一的方法即是Response intercept(Chain chain) throws IOException;

因爲是遞歸調用,那何時結束呢?帶着這個問題,咱們再次看下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, this, eventListener, client.connectTimeoutMillis(),
        client.readTimeoutMillis(), client.writeTimeoutMillis());

    return chain.proceed(originalRequest);
  }
複製代碼

從方法體能夠看到,OkHttp在添加了開發者自定義的Interceptor後,前後添加了RetryAndFollowUpInterceptor...CallServerInterceptor等攔截器。這裏先給出一個結論,攔截器中的責任鏈在CallServerInterceptor這個類中結束。這個結論是怎麼獲得的呢?來看下OkHttp第一個原生攔截器RetryAndFollowUpInterceptor的代碼:前面提到過OKHttp全部的攔截器都實現了Interceptor接口。因此直接來看下Interceptor中核心的方法intercept吧。

RetryAndFollowUpInterceptor攔截器中的intercept方法:

@Override public Response intercept(Chain chain) throws IOException {
    Request request = chain.request();
    RealInterceptorChain realChain = (RealInterceptorChain) chain;
    Call call = realChain.call();
    EventListener eventListener = realChain.eventListener();

    StreamAllocation streamAllocation = new StreamAllocation(client.connectionPool(),
        createAddress(request.url()), call, eventListener, callStackTrace);
    this.streamAllocation = streamAllocation;

    int followUpCount = 0;
    Response priorResponse = null;
    while (true) {
      if (canceled) {
        streamAllocation.release();
        throw new IOException("Canceled");
      }

      Response response;
      boolean releaseConnection = true;
      try {
        response = realChain.proceed(request, streamAllocation, null, null);
        releaseConnection = false;
      } 
      ...
    }
  }
複製代碼

看到沒response = realChain.proceed(request, streamAllocation, null, null);經過這句代碼,開始調用RealInterceptorChainproceed方法,而RealInterceptorChain方法的proceed具體邏輯是取出責任鏈上下一個攔截器並執行該攔截器的intercept方法,這樣造成了一個具體的循環邏輯,責任鏈開始完美的執行。繼續說下責任鏈的結束,因爲CallServerInterceptor這個攔截器是責任鏈的最後一環,咱們看下他的代碼。

CallServerInterceptor的intercept方法

@Override public Response intercept(Chain chain) throws IOException {
  HttpCodec httpCodec = ((RealInterceptorChain) chain).httpStream();
  StreamAllocation streamAllocation = ((RealInterceptorChain) chain).streamAllocation();
  Request request = chain.request();

  long sentRequestMillis = System.currentTimeMillis();
  httpCodec.writeRequestHeaders(request);

  if (HttpMethod.permitsRequestBody(request.method()) && request.body() != null) {
    Sink requestBodyOut = httpCodec.createRequestBody(request, request.body().contentLength());
    BufferedSink bufferedRequestBody = Okio.buffer(requestBodyOut);
    request.body().writeTo(bufferedRequestBody);
    bufferedRequestBody.close();
  }

  httpCodec.finishRequest();

  Response response = httpCodec.readResponseHeaders()
      .request(request)
      .handshake(streamAllocation.connection().handshake())
      .sentRequestAtMillis(sentRequestMillis)
      .receivedResponseAtMillis(System.currentTimeMillis())
      .build();

  if (!forWebSocket || response.code() != 101) {
    response = response.newBuilder()
        .body(httpCodec.openResponseBody(response))
        .build();
  }

  if ("close".equalsIgnoreCase(response.request().header("Connection"))
      || "close".equalsIgnoreCase(response.header("Connection"))) {
    streamAllocation.noNewStreams();
  }

  // 省略部分檢查代碼

  return response;
複製代碼

上述代碼能夠看到,在CallServerInterceptor攔截器中,會調用HttpCodec這個類來完成最終的請求。這裏簡單說下HttpCodec這個接口,在OkHttp中有兩個實現類Http1Codec與Http2Codec,這兩個實現類對應着Http1.x與Http2.x的請求。他們內部的具體請求都依賴於Okio,而Okio內部也是基於Socket的封裝。回到原來的話題,CallServerInterceptor的intercept方法執行後並無再次調用RealInterceptorChain的process方法,而是直接將Response這個響應體返回了。因此攔截器的責任鏈到這裏終止尋找下一環,而且將Response逆向返回。這個時候代碼執行完了攔截器的責任鏈,最終再次返回到RealCall的execute方法中

用張圖總結一下攔截器鏈的調用過程:

攔截器調用流程

至此OkHttp的同步流程就介紹完了。

3.3 異步請求

3.3.1 如何發起異步請求

異步請求的典型代碼

Request request = new Request.Builder()
                .url(url)
                .build();
        client.newCall(request).enqueue(new Callback() {
            @Override
            public void onFailure(Call call, IOException e) {
                //TODO 失敗的回調函數
            }

            @Override
            public void onResponse(Call call, Response response) throws IOException {
                //TODO 成功的回調函數
            }
        });
複製代碼

3.3.2 異步請求涉及到的類

3.3.2.1 Dispatcher類

同步請求的時候我們短暫的遇到過這個類,可是經過請求中這個類不是主角,他僅僅將同步請求的Call存儲到了一個名叫runningSyncCallsArrayDeque中。可是在異步請求中,Dispatcher類仍是承擔了很重要的角色。

public final class Dispatcher {
  private int maxRequests = 64;
  private int maxRequestsPerHost = 5;
  private @Nullable Runnable idleCallback;
  private @Nullable ExecutorService executorService;
  private final Deque<AsyncCall> readyAsyncCalls = new ArrayDeque<>();
  private final Deque<AsyncCall> runningAsyncCalls = new ArrayDeque<>();
  private final Deque<RealCall> runningSyncCalls = new ArrayDeque<>();

  public Dispatcher(ExecutorService executorService) {
    this.executorService = executorService;
  }
  public Dispatcher() {
  }
  public synchronized ExecutorService executorService() {
    if (executorService == null) {
      executorService = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60, TimeUnit.SECONDS,
          new SynchronousQueue<Runnable>(), Util.threadFactory("OkHttp Dispatcher", false));
    }
    return executorService;
  }
  ...
  synchronized void enqueue(AsyncCall call) {
    if (runningAsyncCalls.size() < maxRequests && runningCallsForHost(call) < maxRequestsPerHost) {
      runningAsyncCalls.add(call);
      executorService().execute(call);
    } else {
      readyAsyncCalls.add(call);
    }
  }

  public synchronized void cancelAll() {
    for (AsyncCall call : readyAsyncCalls) {
      call.get().cancel();
    }

    for (AsyncCall call : runningAsyncCalls) {
      call.get().cancel();
    }

    for (RealCall call : runningSyncCalls) {
      call.cancel();
    }
  }
...
}
複製代碼

從上述Dispatcher的源碼來看,該類用final關鍵字修飾,表示該類不能夠被繼承。他的內部有個public synchronized ExecutorService executorService()方法。很明顯這個方法設置了一下ThreadPoolExecutor線程池。異步的任務都將放到這個線程池中執行。這與同步調用直接經過攔截器鏈的方式仍是有明顯差異的。

3.3.2.2 AsyncCall類

這個類繼承自NamedRunnable類,而NamedRunnable類實際上是對Runnable的簡單封裝,主要是爲當前執行的線程設置一下名稱。這裏就不展開對NamedRunnable的介紹了。下面看下AsyncCall的源碼。

final class AsyncCall extends NamedRunnable {
    private final Callback responseCallback;

    AsyncCall(Callback responseCallback) {
      super("OkHttp %s", redactedUrl());
      this.responseCallback = responseCallback;
    }

    String host() {
      return originalRequest.url().host();
    }

    Request request() {
      return originalRequest;
    }

    RealCall get() {
      return RealCall.this;
    }

    @Override protected void execute() {
      boolean signalledCallback = false;
      try {
        Response response = getResponseWithInterceptorChain();
        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 {
          eventListener.callFailed(RealCall.this, e);
          responseCallback.onFailure(RealCall.this, e);
        }
      } finally {
        client.dispatcher().finished(this);
      }
    }
  }
複製代碼

當線程池調用AsyncCall的時候,execute方法會被執行,看下execute的內容,天啦嚕,仍是經過getResponseWithInterceptorChain方法發起的請求,這跟同步調用是同樣同樣滴。

3.3.3 異步請求流程梳理

仍是從3.3.1小節的demo開始,一塊兒來梳理一下異步流程。 開始調用的時候跟同步流程同樣,都是通過client.newCall(request)的調用,很明顯這一步已經走到了RealCall內部。再來看下RealCallenqueue方法。

@Override public void enqueue(Callback responseCallback) {
    synchronized (this) {
      if (executed) throw new IllegalStateException("Already Executed");
      executed = true;
    }
    captureCallStackTrace();
    eventListener.callStart(this);
    client.dispatcher().enqueue(new AsyncCall(responseCallback));
  }
複製代碼

enqueue方法會檢查當前這個Call是否被請求過,若是已經被請求過,則會拋出異常。最後,該方法會將AsyncCall對象直接交給Dispatcher來執行,上面3.3.2對Dispatcher的介紹已經分析過了,Dispatcherenqueue方法會直接用executorService方法初始化的線程池執AsyncCall這個通過封裝的 Runnable。在AsyncCall內部會完成開發者自定義的Callback回調。至此異步流程梳理完畢。

寫在最後

本篇是我第一次寫源碼分析類的博客,寫的可能比較亂。歡迎拍磚。

前段時間參加了掘金的JTalk線下沙龍,滴滴的子成大佬提出的「跳出溫馨期」觀念讓我感覺頗多。畢竟多接觸些新的技術面對本身的發展老是有好處的。

戒驕、戒躁、戒安逸。共勉~


About Me

contact way value
mail weixinjie1993@gmail.com
wechat W2006292
github https://github.com/weixinjie
blog https://juejin.im/user/57673c83207703006bb92bf6
相關文章
相關標籤/搜索