okhttp

链接:https://juejin.cn/post/6887896333685161992

简述:

通过对外提供的OkHttpClient和Request的builder实现基础信息和必要信息的配置,直到封装构建成了RealCall对象(RealCall implement Call)并新建CallBack实例传入realCall.equeue(callback),才真正完成了请求实体的实例化。

之后realCall.enqueue(call)方法的调用才是实际上开始进行请求:先判断是否call已经执行过了(executed = AtomicBoolean()),若未执行则继续

之后由Dispatcher调用enqueue进行判断 并发请求小于64 且同host请求小于5。超过了则将请求放到等待队列中,没超过放到正在执行的队列中,然后调用线程池(默认单例初始化了一个缓存线程池(即无核心线程、无限线程池数量、SynchronousQueue))执行它调度,执行的过程也就是asyncCall的run()方法通过责任链五大拦截器进行层层处理的过程。

RBCCC

RetryAndFollowUp重试重定;Birdge拼接header;Cache判断缓存;Connect建立连接;CallServer发起请求;

参考链接:https://juejin.cn/post/7129306281650683935;待整理

img

拦截器

RetryAndFollowUpInterceptor

处理错误重试和重定向

代码索引

重试重定向拦截器 这个拦截器是一个while(true)的循环,只有请求成功或者重试超过最大次数(20),没有路由供重试时才会退出

请求抛出异常并满足重试条件时才重试,收到3xx,需要重定向时会重新构建请求

重试

  • 只有当请求过程中出现(RouteException或IoExcetpion) 且 recover()返回true才会进行重试,recover()中根据

    1. client配置;

    2. 是否协议异常、是否ssl握手或证书异常;

    3. 是否有更多路由路线;

      等进行判断是否能进行重试。重试次数不限直到遇到recover返回false的异常。否则循环直到成功(如换Route也就是不同的Dns返回的ip重试)

  • 重试次数理论上不做限制,根据重试逻辑直到遇到不可修复的异常

重定向(30x)

代码索引

当请求成功后,若返回的response code 是30x代表重定向,request会循环中被不断重新赋值到重定向新地址,并在下一次循环中执行chain走完整的请求流程。

  • 最大重定向次数为20。
  • 只对300,301,302,303307,308的GET或HEAD方法 执行重定向

//比如https://m.aliexpress.ru/app/newqa/qaDetail.html 就会被重定向3次最后定向到 https://www.aliexpress.com/p/error/404.html

BridgeInterceptor

应用层和网络层的桥接拦截器,主要工作是为请求添加cookie、添加固定的header,比如Host、Content-Length、Content-Type、User-Agent等等,然后保存响应结果的cookie,如果响应使用gzip压缩过,则还需要进行解压。

  • 将http request加工,默认添加header 头字段(Connection:keep-alive;Accept-Encoding:Gzip),再将http response 加工 去掉 header

CacheInterceptor

img

缓存拦截器,如果命中缓存则不会发起网络请求。

  • 缓存默认不开启,需进行配置才能会有磁盘缓存

    1
    /*OkHttp中有一套自己的缓存机制,用户请求服务器时,成功获取响应的同时,会将缓存存储到本地,在下次请求网络时,如果命中并且缓存没有过期也并未修改,OkHttp将会将本地缓存文件读取后返回给客户端,这样便可以大大增加程序的运行效率,减轻服务器的压力.要注意的是 OkHttp中默认是未开启缓存的,若想使用缓存机制,则需要在OkHttpclient.builder中添加cache(newCache(getCacheDir(),1024*1024))来手动添加缓存, getCacheDIr()为缓存存放路径,参数二为文件存储的大小,单位为bit.*/
  • 磁盘缓存数据会将response的header改名xxxx.0 和 response的body改名xxxx.1 ,文件名为请求url的urf8.md5.hex

  • 磁盘缓存实现是DiskLruCache,缓存时会有日志文件journal,其功能与glide基本一致

  • 只缓存 get 请求

journal文件

OkHttp3 使用 DiskLruCache 来实现磁盘缓存管理。DiskLruCache 是一个高效的缓存实现,它使用一个 journal 文件来记录缓存操作。Journal 文件记录了每个缓存条目的以下操作:

  • DIRTY:开始写入缓存条目。
  • CLEAN:完成写入缓存条目,并记录条目的大小。
  • REMOVE:移除一个缓存条目。
  • READ:读取一个缓存条目。

取缓存过程

  • 缓存分为强制缓存和对比缓存
    1. 客户端直接拿数据,若缓存未命中则请求网络并更新数据
    2. 客户端拿数据标识,总是查询网络判断数据是否有效
  • 响应头中包含Cache-Control:max-age,表示缓存过期时间
  • 响应头中有Last-Modified字段标识资源最后被修改时间,客户端下次发起请求时在请求头中会带上If-Modified-Since,服务器比对如果最后修改时间大于该值则返回200,否则304标识缓存有效。
  • 除了用最后修改时间做判断,还可以用资源唯一标识来判断ETag/If-None-Match,响应头包含ETag,再次请求时带上If-None-Match,服务器比对标识是否相同,相同则304,否则200
  • 缓存策略:先判断缓存是否过期,若未过期则直接使用,若过期则发起请求,请求头带唯一标识,服务器回200或304,如果没有唯一标识则请求头带上次修改时间,服务器200或304
  • 在无网环境下即是缓存过期,依然使用缓存,要添加 应用拦截器,重构request修改cache-control字段为FORCE_CACHE:
1
2
3
4
5
6
7
8
9
10
11
12
public class ForceCacheInterceptor implements Interceptor {
@Override
public Response intercept(Chain chain) throws IOException {
Request.Builder builder = chain.request().newBuilder();
if (!NetworkUtils.internetAvailable()) {
builder.cacheControl(CacheControl.FORCE_CACHE);
}

return chain.proceed(builder.build());
}
}
okHttpClient.addInterceptor(new ForceCacheInterceptor());
  • 若服务器不支持header头缓存字段,则可以添加网络拦截器,在CacheInterceptor收到响应之前修改response的header
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class CacheInterceptor implements Interceptor {
@Override
public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
Response response = chain.proceed(request);
Response response1 = response.newBuilder()
.removeHeader("Pragma")
.removeHeader("Cache-Control")
//cache for 30 days
.header("Cache-Control", "max-age=" + 3600 * 24 * 30)
.build();
return response1;
}
}

ConnectInterceptor

连接拦截器,内部会维护一个连接池,负责连接复用、创建连接(三次握手等等)、释放连接以及创建连接上的socket流。

  • 连接拦截器
  • 建立连接及连接上的流
  • 维护连接池,以复用连接
  • 一个物理链接上有多个流(逻辑上的请求响应对),一个物理链接上的多个流是并发的,但有数量限制,一个流上有多个分配,分配是并发的
  • 获取连接流程:
  1. 复用已分配的连接(重定向再次请求)
  2. 无已分配链接,则从链接池那一个新得链接,通过主机名和端口(并不是池中的链接就能复用,除了host之外的字段要都相等,比如dns,协议,代理)
  3. 尝试其他路由再从连接池中获取连接,若找到则进行dns查询
  4. 如果缓存池中没有链接,则新建链接(tcp+tls握手,sockect.connect+connectTls),这是耗时的,过程中可能有连接池可能有新的可用连接 所以再次尝试从连接池获取连接,如果成功则释放刚建立的链接,否则把新建连接入池

连接复用

  • tcp连接建立需要三次握手和四次挥手
  • 连接池实现链接缓存,实现同一地址的链接复用
  • 连接池以队列方式存储链接ArrayDeque,链接池中同一个地址最多维护5个空闲链接,空闲链接最多存活5分钟

连接清理

  • 五分钟定时任务,每五分钟遍历所有链接,并找到其中空闲时间最长的,如果空闲时间超过keep-alive(5分钟),或者空闲链接超过了阈值(5个)则清除这个链接

networkInterceptors

用户自定义拦截器,通常用于监控网络层的数据传输。

  • 网络拦截器

  • 在连接建立完成和发送请求之间

  • 可能不被调用,比如缓存命中,或者多次调用重定向

CallServerInterceptor

请求拦截器,在前置准备工作完成后,真正发起了网络请求。

请求拦截器

将请求和响应分装成 http2 的帧,通过Http2ExchangeCodec(内部通过okio实现io)

1 写入请求头 - 2 写入请求体 - 3 读取响应头 - 4 读取响应体 如果响应头中 Connection:close,则在当前链接上设置标志位,表示该链接不能再被复用

  • RetryAndFollowUpInterceptor是在一个死循环中进行最大次数为20(MAX_FOLLOW_UPS)的重试(超过抛ProtocolException(“Too many follow-up requests: $followUpCount”)),

    每次重试根据不同的responseCode进行补偿【比如30x(除304外,304为协商缓存)进行重定向,503】,对于不必要的情况也会直接终止(HTTP_CLIENT_TIMEOUT 408)。

  • CacheInterceptor是当请求发起时,先检查是否命中缓存(强制缓存或协商缓存),无命中缓存或缓存失效,则将请求向下传递给下一个拦截器。

    获得了响应数据后,检查响应头的Cache-Control,Expires,no-store不存储,immutable等判断是否写入缓存

img

用了 责任链设计模式 ,它将请求一层一层向下传,直到有一层能够得到Resposne就停止向下传递,并 Response 向上面的拦截器传递,然后各个拦截器会对 respone 进行一些处理,最后会传到 RealCall 类中通过 execute 来得到 Response 。

img

代码解析:

一:构建realCall

1、2、3步其实都是基础信息和必要信息的配置,基本用构建者模式来完成,直到最后构建成了realCall,才真正完成了请求实体,之后realCall.enqueue方法的调用才是实际上开始为请求调度线程,开始通过责任链一节节的处理请求发起。

二:封装CallBack成AsyncCall,通过dispatcher.equeue调度

在call.enqueue(new Callback(){ … }) 执行之后,首先做的是

1. 调用RealCall.enqueue()方法,判断当前call是否已经被执行过了,被执行过了,就抛出异常,如果没有执行过,就先将callback封装成AsyncCall,然后调用dispatcher.enqueue()方法(dispatcher调度器,在okHttpClient里面创建的)

2. 在dispatcher.enqueue()方法中,判断当前正在执行的请求数及当前网络请求的主机数是否超过了最大值。要是超过了最大值,就将请求放到等待队列中,要是没超过,就放到正在执行的队列中,然后调用线程池执行它。

三 .realcall的异步请求的执行

通过getResponseWithInterceptorChain()责任链处理并最终发起请求接收处理回调

1 executorService.execute

吃过了午饭,继续分析源码,承接上文,我们分析到了,如果符合条件,就用线程池去执行它,也就是这句

1
executorService().execute(call);

看一下我们的线程池

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public synchronized ExecutorService executorService() {
if (executorService == null) {
1.executorService = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60, TimeUnit.SECONDS, new SynchronousQueue<Runnable>(), Util.threadFactory("OkHttp Dispatcher", false));
}
return executorService;
}


//实际上就是一个缓存线程池,但是由于dispatcher的enqueue中已经对请求数做了限定,故不会由太大的性能负担。
//无核心线程、MAX_VALUE个最大线程、闲置60秒回收。
//Excutors.class
public static ExecutorService newCachedThreadPool() {
return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
60L, TimeUnit.SECONDS,
new SynchronousQueue<Runnable>());
}

无核心线程、MAX_VALUE个最大线程、非核心线程闲置60秒回收。于dispatcher的enqueue中已经对请求数做了限定,故不会由太大的性能负担。

好,继续再看下线程池executorService.execute(call)方法 它会执行里面call方法的run()方法,也就是AsyncCall的run方法,实际上是调用了execute(),该方法由子类实现,也就是调用了AsyncCall.execute()

简单来说,就是executorService.execute(call) -> AsyncCall.run() -> AsyncCall.execute() 看到这个写法,内心中就想吐槽一句话,真鸡儿秀! 来继续看下,AsyncCall.execute()

2 AsyncCall.execute()

看源码 文件位置:realcall.java

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
@Override
protected void execute() {
boolean signalledCallback = false;
try {
1.Response response = getResponseWithInterceptorChain();
2.if (retryAndFollowUpInterceptor.isCanceled()) {
signalledCallback = true;
responseCallback.onFailure(RealCall.this, new IOException("Canceled"));
} else {
signalledCallback = true;
3.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 {
4.client.dispatcher().finished(this);
}
}
  1. 执行拦截器链,返回Response
  2. 判断拦截器链中的重定向拦截器是否已经取消了,如果取消了,就执行responseCallback.onFailure() ,这个也就是我们在外边在第3步,传过来的回调方法Callback()中的onFailure()方法。
  3. 如果没取消,则走onResponse也就是Callback()中的onResponse()方法。返回结果。当然这里都是在子线程里面的。
  4. 这句其实就是调用了,dispatcher方法中的finished方法。下面我们看下

3 dispatcher.finished()

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
void finished(RealCall call) {
finished(runningSyncCalls, call, false);
}

private <T> void finished(Deque<T> calls, T call, boolean promoteCalls) {
int runningCallsCount;
Runnable idleCallback;
synchronized (this) {
1.if (!calls.remove(call)) throw new AssertionError("Call wasn't in-flight!");
2.if (promoteCalls) promoteCalls();
3.runningCallsCount = runningCallsCount();
idleCallback = this.idleCallback;
}

if (runningCallsCount == 0 && idleCallback != null) {
idleCallback.run();
}
}
  1. 将该请求从正在执行的任务队列里面删除
  2. 调用promoteCalls() 调整请求队列
  3. 重新计算请求数量

4 dispatcher.promoteCalls()

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
private void promoteCalls() {
if (runningAsyncCalls.size() >= maxRequests) return; // Already running max capacity.
if (readyAsyncCalls.isEmpty()) return; // No ready calls to promote.

for (Iterator<AsyncCall> i = readyAsyncCalls.iterator(); i.hasNext(); ) {
AsyncCall call = i.next();

if (runningCallsForHost(call) < maxRequestsPerHost) {
i.remove();
runningAsyncCalls.add(call);
executorService().execute(call);
}

if (runningAsyncCalls.size() >= maxRequests) return; // Reached max capacity.
}
}

很简单,这里无非就是遍历等待队列中的请求,然后加入到执行请求队列中,直到并发数和当前网络请求的主机数达到上限。

至此,okhttp的异步已经分析完毕了

面试题 1.什么是dispatcher? dispatcher作用是为维护请求的状态,并维护一个线程池。用于执行请求。

小伙子,是不是很简单呀?是不是已经完了?你想多了,来分析最核心的一部分

细说 getResponseWithInterceptorChain()

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36

Response getResponseWithInterceptorChain() throws IOException {
// 添加拦截器,责任链模式
List<Interceptor> interceptors = new ArrayList<>();

// 在配置okhttpClient 时设置的intercept 由用户自己设置
interceptors.addAll(client.interceptors());

// 负责处理失败后的重试与重定向
interceptors.add(retryAndFollowUpInterceptor);

/** 负责把用户构造的请求转换为发送到服务器的请求 、把服务器返回的响应转换为用户友好的响应 处理 配置请求头等信息.
从应用程序代码到网络代码的桥梁。首先,它根据用户请求构建网络请求。然后它继续呼叫网络。最后,它根据网络响应构建用户响应。
*/
interceptors.add(new BridgeInterceptor(client.cookieJar()));

// 处理 缓存配置 根据条件(存在响应缓存并被设置为不变的或者响应在有效期内)返回缓存响应
// 设置请求头(If-None-Match、If-Modified-Since等) 服务器可能返回304(未修改)
// 可配置用户自己设置的缓存拦截器
interceptors.add(new CacheInterceptor(client.internalCache()));

// 连接服务器 负责和服务器建立连接 这里才是真正的请求网络
interceptors.add(new ConnectInterceptor(client));
if (!forWebSocket) {
interceptors.addAll(client.networkInterceptors());
}

// 执行流操作(写出请求体、获得响应数据) 负责向服务器发送请求数据、从服务器读取响应数据
// 进行http请求报文的封装与请求报文的解析
interceptors.add(new CallServerInterceptor(forWebSocket));

// 责任链,将上述的拦截器添加到责任链里面
Interceptor.Chain chain = new RealInterceptorChain(interceptors, null, null, null, 0, originalRequest);
return chain.proceed(originalRequest);
}

拦截器 作用
Interceptor应用拦截器 拿到的是原始请求,可以添加一些自定义header、通用参数、参数加密、网关接入等等。
RetryAndFollowUpInterceptor 处理错误重试和重定向
BridgeInterceptor 应用层和网络层的桥接拦截器,主要工作是为请求添加cookie、添加固定的header,比如Host、Content-Length、Content-Type、User-Agent等等,然后保存响应结果的cookie,如果响应使用gzip压缩过,则还需要进行解压。
CacheInterceptor 缓存拦截器,如果命中缓存则不会发起网络请求。
ConnectInterceptor 连接拦截器,内部会维护一个连接池,负责连接复用、创建连接(三次握手等等)、释放连接以及创建连接上的socket流。
networkInterceptors(网络拦截器) 用户自定义拦截器,通常用于监控网络层的数据传输。
CallServerInterceptor 请求拦截器,在前置准备工作完成后,真正发起了网络请求。

Other

addInterceptor与addNetworkInterceptor的区别

二者通常的叫法为应用拦截器和网络拦截器,从整个责任链路来看,应用拦截器是最先执行的拦截器,也就是用户自己设置request属性后的原始请求,而网络拦截器位于ConnectInterceptor和CallServerInterceptor之间,此时网络链路已经准备好,只等待发送请求数据。

  1. 首先,应用拦截器在RetryAndFollowUpInterceptor和CacheInterceptor之前,所以一旦发生错误重试或者网络重定向,网络拦截器可能执行多次,因为相当于进行了二次请求,但是应用拦截器永远只会触发一次。另外如果在CacheInterceptor中命中了缓存就不需要走网络请求了,因此会存在短路网络拦截器的情况。
  2. 其次,如上文提到除了CallServerInterceptor,每个拦截器都应该至少调用一次realChain.proceed方法。实际上在应用拦截器这层可以多次调用proceed方法(本地异常重试)或者不调用proceed方法(中断),但是网络拦截器这层连接已经准备好,可且仅可调用一次proceed方法。
  3. 最后,从使用场景看,应用拦截器因为只会调用一次,通常用于统计客户端的网络请求发起情况;而网络拦截器一次调用代表了一定会发起一次网络通信,因此通常可用于统计网络链路上传输的数据。

网络缓存机制CacheInterceptor

这里的缓存是指基于Http网络协议的数据缓存策略,侧重点在客户端缓存,所以我们要先来复习一下Http协议如何根据请求和响应头来标识缓存的可用性。

提到缓存,就必须要聊聊缓存的有效性、有效期。

HTTP缓存原理

在HTTP 1.0时代,响应使用Expires头标识缓存的有效期,其值是一个绝对时间,比如Expires:Thu,31 Dec 2020 23:59:59 GMT。当客户端再次发出网络请求时可比较当前时间 和上次响应的expires时间进行比较,来决定是使用缓存还是发起新的请求。

使用Expires头最大的问题是它依赖客户端的本地时间,如果用户自己修改了本地时间,就会导致无法准确的判断缓存是否过期。

因此,从HTTP 1.1 开始使用Cache-Control头表示缓存状态,它的优先级高于Expires,常见的取值为下面的一个或多个。

  • private,默认值,标识那些私有的业务逻辑数据,比如根据用户行为下发的推荐数据。该模式下网络链路中的代理服务器等节点不应该缓存这部分数据,因为没有实际意义。
  • public 与private相反,public用于标识那些通用的业务数据,比如获取新闻列表,所有人看到的都是同一份数据,因此客户端、代理服务器都可以缓存。
  • no-cache 可进行缓存,但在客户端使用缓存前必须要去服务端进行缓存资源有效性的验证,即下文的对比缓存部分,我们稍后介绍。
  • max-age 表示缓存时长单位为秒,指一个时间段,比如一年,通常用于不经常变化的静态资源。
  • no-store 任何节点禁止使用缓存。

强制缓存

在上述缓存头规约基础之上,强制缓存是指网络请求响应header标识了Expires或Cache-Control带了max-age信息,而此时客户端计算缓存并未过期,则可以直接使用本地缓存内容,而不用真正的发起一次网络请求。

协商缓存

强制缓存最大的问题是,一旦服务端资源有更新,直到缓存时间截止前,客户端无法获取到最新的资源(除非请求时手动添加no-store头),另外大部分情况下服务器的资源无法直接确定缓存失效时间,所以使用对比缓存更灵活一些。

使用Last-Modify / If-Modify-Since头或者ETag/If-None-Match实现协商缓存,具体方法是服务端响应头添加Last-Modify头标识资源的最后修改时间,单位为秒,当客户端再次发起请求时添加If-Modify-Since头并赋值为上次请求拿到的Last-Modify头的值。

服务端收到请求后自行判断缓存资源是否仍然有效,如果有效则返回状态码304同时body体为空,否则下发最新的资源数据。客户端如果发现状态码是304,则取出本地的缓存数据作为响应。

使用这套方案有一个问题,那就是资源文件使用最后修改时间有一定的局限性:

  1. Last-Modify单位为秒,如果某些文件在一秒内被修改则并不能准确的标识修改时间。
  2. 资源修改时间并不能作为资源是否修改的唯一依据,比如资源文件是Daily Build的,每天都会生成新的,但是其实际内容可能并未改变。

因此,HTTP 还提供了另外一组头信息来处理缓存,ETag/If-None-Match。流程与Last-Modify一样,只是把服务端响应的头变成Last-Modify,客户端发出的头变成If-None-Match。ETag是资源的唯一标识符 ,服务端资源变化一定会导致ETag变化。具体的生成方式有服务端控制,场景的影响因素包括,文件最终修改时间、文件大小、文件编号等等。

OKHttp的缓存实现

上面讲了这么多,实际上OKHttp就是将上述流程用代码实现了一下,即:

  1. 第一次拿到响应后根据头信息决定是否缓存。
  2. 下次请求时判断是否存在本地缓存,是否需要使用对比缓存、封装请求头信息等等。
  3. 如果缓存失效或者需要对比缓存则发出网络请求,否则使用本地缓存。

OKHttp内部使用Okio来实现缓存文件的读写。

缓存文件分为CleanFiles和DirtyFiles,CleanFiles用于读,DirtyFiles用于写,他们都是数组,长度为2,表示两个文件,即缓存的请求头和请求体;同时记录了缓存的操作日志,记录在journalFile中。

开启缓存需要在OkHttpClient创建时设置一个Cache对象,并指定缓存目录和缓存大小,缓存系统内部使用LRU作为缓存的淘汰算法。

1
2
3
4
5
6
7
## Cache.kt
class Cache internal constructor(
directory: File,
maxSize: Long,
fileSystem: FileSystem
): Closeable, Flushable
复制代码

OkHttp早期的版本有个一个InternalCache接口,支持自定义实现缓存,但到了4.x的版本后删减了InternalCache,Cache类又为final的,相当于关闭了扩展功能。

具体源码实现都在CacheInterceptor类中,大家可以自行查阅。

通过OkHttpClient设置缓存是全局状态的,如果我们想对某个特定的request使用或禁用缓存,可以通过CacheControl相关的API实现:

1
2
3
4
5
6
//禁用缓存
Request request = new Request.Builder()
.cacheControl(new CacheControl.Builder().noCache().build())
.url("http://publicobject.com/helloworld.txt")
.build();
复制代码

OKHttp不支持的缓存情况

最后需要注意的一点是,OKHttp默认只支持get请求的缓存。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
# okhttp3.Cache.java
@Nullable CacheRequest put(Response response) {
String requestMethod = response.request().method();
...
//缓存仅支持GET请求
if (!requestMethod.equals("GET")) {
// Don't cache non-GET responses. We're technically allowed to cache
// HEAD requests and some POST requests, but the complexity of doing
// so is high and the benefit is low.
return null;
}

//对于vary头的值为*的情况,统一不缓存
if (HttpHeaders.hasVaryAll(response)) {
return null;
}
...
}
复制代码

这是当网络请求响应后,准备进行缓存时的逻辑代码,当返回null时表示不缓存。从代码注释中不难看出,我们从技术上可以缓存method为HEAD和部分POST请求,但实现起来的复杂性很高而收益甚微。这本质上是由各个method的使用场景决定的。

我们先来看看常见的method类型及其用途。

  • GET 请求资源,参数都在URL中。
  • HEAD 与GET基本一致,只不过其不返回消息体,通常用于速度或带宽优先的场景,比如检查资源有效性,可访问性等等。
  • POST 提交表单,修改数据,参数在body中。
  • PUT 与POST基本一致,最大不同为PUT是幂等的。
  • DELETE 删除指定资源。

可以看到对于标准的RESTful请求,GET就是用来获取数据,最适合使用缓存,而对于数据的其他操作缓存意义不大或者根本不需要缓存。

也是基于此在仅支持GET请求的条件下,OKHTTP使用request URL作为缓存的key(当然还会经过一系列摘要算法)。

OkHttp的线程池

OkHttp中的线程池是定义在分发器中的,即定义在Dispatcher

1
2
3
4
5
6
7
public synchronized ExecutorService executorService() {
if (executorService == null) {
executorService = new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60, TimeUnit.SECONDS,
new SynchronousQueue<>(), Util.threadFactory("OkHttp Dispatcher", false));
}
return executorService;
}

高并发,最大吞吐量。SynchronousQueue队列是无容量队列,
在OkHttp中,配置的线程池的核心线程数为0,最大线程数为Integer.MAX_VALUE,线程的存活时间为60s,采用的队列是SynchronousQueue。

  1. okhttp 默认同时支持 64 个异步请求(不考虑同步请求),一个 host 同时最多请求 5 个
  2. okhttp 内部的线程池都是 CacheThreadPool:核心线程数为 0,非核心线程数无限,永远添加不到等待队列中
  3. okhttpClient 如果不单例,会出现 oom:因为大量的 Dispatcher 对象,不同的对象会使用不同的线程去发起网络请求,从而导致线程过多,OOM

相关代码

RetryAndFollowUpInterceptor

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
try{
response = realChain.proceed(request, streamAllocation, null, null);
//对执行之后拦截器过程中可能出现的异常进行捕获,重试的机制主要在异常的捕获中来判断
}catch(RouteException e){ //出现路由连接失败
//通过recover()方法来判断是否有必要进行重试,若是一些不可更改的错误,则直接抛出异常,终止此次请求,否则执行下一次循环
if (!recover(e.getLastConnectException(), streamAllocation, false, request)) {
throw e.getLastConnectException();
}
continue; //后续代码将不在执行,进入下一循环
}catch (IOException e) {//远程连接后,出现双方通信管道(io)数据交互异常
boolean requestSendStarted = !(e instanceof ConnectionShutdownException); //ConnectionShutdownException只有http2才会被抛出,若为http1则一定为true
if (!recover(e, streamAllocation, requestSendStarted, request))
throw e; //recover返回false,抛出异常并结束
releaseConnection = false;
continue;
}finally { //可以理解为若异常非以上两种异常情况,将直接回收此连接,不会在重试
if (releaseConnection) {
streamAllocation.streamFailed(null);
streamAllocation.release();
}
}

private boolean recover(IOException e, StreamAllocation streamAllocation,
boolean requestSendStarted, Request userRequest) {
//客户端初始化OkhttpClient时配置不允许重试,
if (!client.retryOnConnectionFailure()) return false;
//若为RouteException,条件1参数为false
//若为IOException,之前说过http2才会抛ConnectionShutdownException异常,那么满足条件2--body只允许发送一次
//UnrepeatableRequestBody为一个接口,相当于一个标记,body实现此接口后,连接将不会被重试
if (requestSendStarted && userRequest.body() instanceof UnrepeatableRequestBody) return false;
//方法判断是否为不可修复异常
if (!isRecoverable(e, requestSendStarted)) return false;
//查找是否有更多的路由路线进行连接
if (!streamAllocation.hasMoreRoutes()) return false;

return true; //以上条件都不满足,具备重试条件,可重试
}

private boolean isRecoverable(IOException e, boolean requestSendStarted) {
//协议异常,不可修复
if (e instanceof ProtocolException) {
return false;
}
//如果是连接超时则可以重试
if (e instanceof InterruptedIOException) {
return e instanceof SocketTimeoutException && !requestSendStarted;
}
//SSL握手协议,证书异常,不能重试
if (e instanceof SSLHandshakeException) {
if (e.getCause() instanceof CertificateException) {
return false;
}
}
//SSL握手验证异常,不可重试
if (e instanceof SSLPeerUnverifiedException) {
// e.g. a certificate pinning error.
return false;
}
}

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
private Request followUpRequest(Response userResponse, Route route) throws IOException {
int responseCode = userResponse.code();

final String method = userResponse.request().method();
switch (responseCode) {
case HTTP_PROXY_AUTH: //407 客户端使用HTTP代理服务器,要求代理服务器授权后重新请求
Proxy selectedProxy = route != null
? route.proxy()
: client.proxy();
return client.proxyAuthenticator().authenticate(route, userResponse);
// 401 验证身份,服务器需要验证使用者的身份,客户端需要在header中加入Authorization
case HTTP_UNAUTHORIZED:
return client.authenticator().authenticate(route, userResponse);

case HTTP_PERM_REDIRECT: //308 永久重定向
case HTTP_TEMP_REDIRECT: //307 临时重定向
//请求方法非get或者非head,不会重定向
if (!method.equals("GET") && !method.equals("HEAD")) {
return null;
}
// 重定向
case HTTP_MULT_CHOICE: //300
case HTTP_MOVED_PERM: //301
case HTTP_MOVED_TEMP: //302
case HTTP_SEE_OTHER: //303
//客户端不允许重定向,返回空
if (!client.followRedirects()) return null;
// 服务端响应的新地址在响应头的Location字段中
String location = userResponse.header("Location");
if (location == null) return null;
// 将旧地址替换,返回新url对象
HttpUrl url = userResponse.request().url().resolve(location);

// 若返回新地址协议不受支持,可能是系统或框架不支持,取不到url对象
if (url == null) return null;

// 如果地址是http与https的切换,确定客户端是否允许
boolean sameScheme = url.scheme().equals(userResponse.request().url().scheme());
if (!sameScheme && !client.followSslRedirects()) return null;

Request.Builder requestBuilder = userResponse.request().newBuilder();

// 重定向请求中,只有是"PROPFIND""请求才能拥有请求体,否则无论是"POST"还是其他都要改为"GET"请求
if (HttpMethod.permitsRequestBody(method)) {//请求非get和header
final boolean maintainBody = HttpMethod.redirectsWithBody(method); //请求为PROPFIND
if (HttpMethod.redirectsToGet(method)) { //如果非PROPFIND
requestBuilder.method("GET", null); //将请求方式改为get,body设为空
} else {
RequestBody requestBody = maintainBody ? userResponse.request().body() : null; //PROPFIND则添加body
requestBuilder.method(method, requestBody);
}
// 若不是PROPFIND请求,body为null,所以要将请求头中有关body头信息删除
if (!maintainBody) {
requestBuilder.removeHeader("Transfer-Encoding");
requestBuilder.removeHeader("Content-Length");
requestBuilder.removeHeader("Content-Type");
}
}

// 当跨主机重定向时,删除请求头Authorization身份验证信息
if (!sameConnection(userResponse, url)) {
requestBuilder.removeHeader("Authorization");
}

return requestBuilder.url(url).build();

// 408 客户端请求超时,4开头已经不是重定向了,连接已失败
case HTTP_CLIENT_TIMEOUT:
// 判断用户是否允许重试
if (!client.retryOnConnectionFailure()) {
return null;
}
// 此处的判断似曾相识,在重试recover()条件判断中也有此条件,有该接口标记的body,将不会被重试
if (userResponse.request().body() instanceof UnrepeatableRequestBody) {
return null;
}
//priorResponse为上一次请求保存的响应,若上一次的响应也是408重试,放弃
if (userResponse.priorResponse() != null
&& userResponse.priorResponse().code() == HTTP_CLIENT_TIMEOUT) {
return null;
}
// 响应中有Retry-After属性,代表多久后重试,则直接返回null
if (retryAfter(userResponse, 0) > 0) {
return null;
}

return userResponse.request();
// 503 服务器不可用,也代表连接失败
case HTTP_UNAVAILABLE:
if (userResponse.priorResponse() != null
&& userResponse.priorResponse().code() == HTTP_UNAVAILABLE) {
return null;
}

if (retryAfter(userResponse, Integer.MAX_VALUE) == 0) {
return userResponse.request();
}

return null;

default:
return null;
}
}

Author

white crow

Posted on

2021-04-07

Updated on

2024-12-04

Licensed under