OkHttp源码(二)

在RealCall的execute()和AsyncCall的execute()中都调用了getResponseWithInterceptorChain(),这个方法返回了Response,
从request变成了response,这其中发生了什么?拦截器和拦截器链是什么?

Interceptor是什么

Interceptors are a powerful mechanism that can monitor, rewrite, and retry calls.

OkHttp官网说拦截器是一种强大的机制,可以监视、重写和重试调用,它记录传出的请求和传入的响应。

OkHttp源码(二)_第1张图片
image

看上面那张图,有三种拦截器,应用拦截器、网络拦截器和OkHttp core核心代码中的拦截器。可以这样理解,application生成request,通过应用拦截器对其做一系列处理后给OkHttp core中的拦截器,之后通过网络拦截器再做处理,最终访问到服务端返回response,经过网络拦截器处理response,再经过OkHttp core的拦截器处理,最终交给应用拦截器处理,返回给应用。

我们回过头来看getResponseWithInterceptorChain()中初始拦截器的队列的代码:

Response getResponseWithInterceptorChain() throws IOException {
    // Build a full stack of interceptors.
    List 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);
}

client.interceptors()、client.networkInterceptors()这两个队列就是在client中初始化的应用拦截器队列和网络拦截器队列。拦截器队列加入的顺序是,应用拦截器、retryAndFollowUpInterceptor、BridgeInterceptor、CacheInterceptor、ConnectIntercepto、网络拦截器(非webSocket)、CallServerInterceptor发送请求。这样就和上面分析的过程一样了。

画张图更好的理解一下过程:

OkHttp源码(二)_第2张图片

request和response都经过了每一个拦截器,每个拦截器有每个拦截器的作用,这里用到了责任链模式。

那这样的链式调用是如何实现的呢?

InterceptorChain拦截器链

在getResponseWithInterceptorChain()中,添加完拦截器后创建了RealInterceptorChain对象。

Interceptor.Chain chain = new RealInterceptorChain(
        interceptors, null, null, null, 0,
        originalRequest, this, eventListener, 
        client.connectTimeoutMillis(), 
        client.readTimeoutMillis(), 
        client.writeTimeoutMillis()
);
public RealInterceptorChain(
    List interceptors, 
    StreamAllocation streamAllocation,
    HttpCodec httpCodec, 
    RealConnection connection, 
    int index, 
    Request request, 
    Call call,
    EventListener eventListener, 
    int connectTimeout, 
    int readTimeout, 
    int writeTimeout
)

我们先注意这几个参数,interceptors为先前创建好的拦截器集合,index为当前拦截器在interceptors的下标,request传入当前realReques。构造器没有做其他的工作,只是将参数赋值给成员变量。

在创建好RealInterceptorChain后调用了proceed()。

@Override public Response proceed(Request request) throws IOException {
    return proceed(request, streamAllocation, httpCodec, connection);
}

调用了四个参数的proceed()

public Response proceed(Request request, StreamAllocation streamAllocation, HttpCodec httpCodec,
      RealConnection connection) throws IOException {
    
    //参数的判断...

    // 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);

    //对response等的判断...

    return response;
}

proceed()中除了一些对参数的判断和对response的判断外剩下部分就是实现链式调用的重点了。首先看到又创建了一个RealInterceptorChain对象,除了index不一样,其他参数和自己都是一样的,index加了1,就说明这一个拦截器链的起始是比自身起始后一位的,移到了拦截器队列的下一个拦截器。

Interceptor interceptor = interceptors.get(index);
Response response = interceptor.intercept(next);

先获取了拦截器链当前的拦截器,之后调用其intercept()方法,传入了下一个拦截器链。来看intercept()的实现,interceptor是一个接口,有五个实现类分别是上面说到的五个OkHttp默认拦截器。我们先来看RetryAndFollowUpInterceptor的intercept(),因为它是OkHttp core的第一个拦截器。把处理相关的代码都省略了,我们先看链式调用的实现。

先是将chain强制转换为RealInterceptorChain,之后调用了它的proceed(),回到了最初的RealInterceptorChain的proceed(),而proceed()中又有对下一个拦截器链intercept()的调用。

@Override public Response intercept(Chain chain) throws IOException {
    //...
    RealInterceptorChain realChain = (RealInterceptorChain) chain;
    //...
    response = realChain.proceed(request, streamAllocation, null, null);
    //...
}

总结下来就是RealInterceptorChain的proceed()和index控制的不同拦截器实现的intercpt()的递归调用实现了链式处理。

OkHttp源码(二)_第3张图片

现在我们大概知道了拦截器的操作流程,接下来该分析每一个拦截器的功能了。

你可能感兴趣的:(OkHttp源码(二))