最近看了OkHttp(3.11.0)的源码,想总结下本身对OkHttp的认识,加深印象,若有不对的地方欢迎各位大佬指正。安全
一、OkHttp简单的异步请求。
OkHttpClient okHttpClient = new OkHttpClient();
Request request = new Request.Builder().url(url).build();
Call call = okHttpClient.newCall(request);
call.enqueue(new Callback() {
@Override
public void onResponse(Call call, Response response) throws IOException {
}
@Override
public void onFailure(Call call, IOException e) {
}
});
复制代码
OkHttp的请求分四步:bash
- 1.建立OkHttpClient对象
- 2.建立Request对象封装请求url以及请求参数
- 3.经过OkhttpClient对象和Request对象获得Call对象
- 4.最终经过Call对象来执行请求。
二、经过跟踪Call的enqueue方法探索OkHttp的请求过程
这里咱们会发现Call是一个接口,真正实现enqueue方法的是RealCall对象,因此这里展现的也是RealCall的enqueue方法。前面的同步代码块主要是防止enqueue方法屡次执行的,99行主要是将当前Call对象加入到事件监听里面。这里核心的是100行。100行又调用了Dispatcher(Okhttp任务调度类)的equeue方法。而后用Callback对象建立了AsyncCall对象。
三、探索AsyncCall类的真面目
这里我发现AsyncCall继承于NamedRunnable类。
而NamedRunnable类又是一个实现Runnable接口的抽象类,并在run方法中执行了execute方法。而AysbcCall又继承了NamedRunnable,因此AysncCall也是一个Runnable,回过头再看AysncCall的execute方法,不难发现这就是AysncCall类的核心。
四、回过头咱们看下Dispatcher的enqueue方法
这是一个同步的,线程安全的方法。runningAsyncCalls是一个正在执行的存储AysncCall的队列,readyAsyncCalls是一个等待执行的存储AysncCall的队列。maxRequests在Okhttp中定义的正在执行最大请求数是64,支持修改。maxRequestsPerHost在Okhttp中定义的同一主机请求最大数为5。因此这段的含义是判断当前正在执行的请求数和与当前同一主机数是否超过了最大限制,若是超过了限制把AysncCall加入到等待执行队列,不然加入到正在执行队列,并用线程池执行AysncCall,这时会调用AysncCall的execute方法
五、这时候再看AsyncCall的execute方法
这里的147行是okhttp的核心经过一系列的拦截器生成最后的返回Response。以后的148行到162行都是判断这次请求是否成功经过传入的CallBack返回请求结果。164行咱们又见到了Dispatcher对象,在这里调用了他的finished方法,将当前AsyncCall对象传了进去。这里咱们进去看下最后调用的finished方法。
这里202行将当前的AysncCall对象从正在执行的请求队列runningAsyncCalls中移除了。而后咱们主要点进去203行看promoteCalls方法。
这里 先是判断了正在执行的请求数是否大于最大请求数,而且等待执行请求队列不为空。而后遍历等待执行请求队列,取出其中的AsyncCall对象,判断当前正在执行队列中与这个AsyncCall对象相同的主机数有没有达到最大值。若是没有的话则把AsyncCall对象从等待执行队列中移除,加入到正在执行的队列,并用线程池执行这个AsyncCall对象。
六、总结Okhttp内容进行一次网络请求流程。
- 一、ReallCall对象的enqueue方法建立一个内部类AysncCall对象。
- 二、调用Diapatcher的enqueque方法判断是否知足执行此AysncCall条件,知足则将其加入到正在执行的请求队列中,并开始执行,不知足则将其加入到等待执行的请求队列中。
- 三、AysncCall的execute方法经过OkHttp一系列的拦截器生成此次请求的Response对象,经过CallBack对象将请求结果传递给调用者。
- 四、调用Diapatcher的finished的方法将此AysncCall对象从正在执行的队列中移除,遍历等待执行的队列,从中取出知足执行条件的AysncCall对象,将其添加到正在执行的队列中,并执行此AysncCall对象。如此这样反复循环。
- 五、这里总结的是一次请求是怎么被分配执行的,而请求生成结果的核心仍是在AysncCall的execute方法中的第147行,经过拦截器链生成返回结果。