OkHttp源码学习笔记(一)请求流程分析

205 阅读4分钟

最近看了OkHttp(3.11.0)的源码,想总结下自己对OkHttp的认识,加深印象,如有不对的地方欢迎各位大佬指正。

1、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的请求分四步:

  • 1.创建OkHttpClient对象
  • 2.创建Request对象封装请求url以及请求参数
  • 3.通过OkhttpClient对象和Request对象得到Call对象
  • 4.最终通过Call对象来执行请求。

2、通过跟踪Call的enqueue方法探索OkHttp的请求过程

image.png
这里我们会发现Call是一个接口,真正实现enqueue方法的是RealCall对象,所以这里展示的也是RealCall的enqueue方法。前面的同步代码块主要是防止enqueue方法多次执行的,99行主要是将当前Call对象加入到事件监听里面。这里核心的是100行。100行又调用了Dispatcher(Okhttp任务调度类)的equeue方法。然后用Callback对象创建了AsyncCall对象。

3、探索AsyncCall类的真面目

image.png
这里我发现AsyncCall继承于NamedRunnable类。
image.png
而NamedRunnable类又是一个实现Runnable接口的抽象类,并在run方法中执行了execute方法。而AysbcCall又继承了NamedRunnable,所以AysncCall也是一个Runnable,回过头再看AysncCall的execute方法,不难发现这就是AysncCall类的核心。

4、回过头我们看下Dispatcher的enqueue方法

image.png
这是一个同步的,线程安全的方法。runningAsyncCalls是一个正在执行的存储AysncCall的队列,readyAsyncCalls是一个等待执行的存储AysncCall的队列。maxRequests在Okhttp中定义的正在执行最大请求数是64,支持修改。maxRequestsPerHost在Okhttp中定义的同一主机请求最大数为5。所以这段的含义是判断当前正在执行的请求数和与当前同一主机数是否超过了最大限制,如果超过了限制把AysncCall加入到等待执行队列,否则加入到正在执行队列,并用线程池执行AysncCall,这时会调用AysncCall的execute方法

5、这时候再看AsyncCall的execute方法

image.png
这里的147行是okhttp的核心通过一系列的拦截器生成最后的返回Response。之后的148行到162行都是判断此次请求是否成功通过传入的CallBack返回请求结果。164行我们又见到了Dispatcher对象,在这里调用了他的finished方法,将当前AsyncCall对象传了进去。这里我们进去看下最后调用的finished方法。
image.png
image.png
这里202行将当前的AysncCall对象从正在执行的请求队列runningAsyncCalls中移除了。然后我们主要点进去203行看promoteCalls方法。
image.png
这里 先是判断了正在执行的请求数是否大于最大请求数,并且等待执行请求队列不为空。然后遍历等待执行请求队列,取出其中的AsyncCall对象,判断当前正在执行队列中与这个AsyncCall对象相同的主机数有没有达到最大值。如果没有的话则把AsyncCall对象从等待执行队列中移除,加入到正在执行的队列,并用线程池执行这个AsyncCall对象。

6、总结Okhttp内容进行一次网络请求流程。

  • 1、ReallCall对象的enqueue方法创建一个内部类AysncCall对象。
  • 2、调用Diapatcher的enqueque方法判断是否满足执行此AysncCall条件,满足则将其加入到正在执行的请求队列中,并开始执行,不满足则将其加入到等待执行的请求队列中。
  • 3、AysncCall的execute方法通过OkHttp一系列的拦截器生成这次请求的Response对象,通过CallBack对象将请求结果传递给调用者。
  • 4、调用Diapatcher的finished的方法将此AysncCall对象从正在执行的队列中移除,遍历等待执行的队列,从中取出满足执行条件的AysncCall对象,将其添加到正在执行的队列中,并执行此AysncCall对象。如此这样反复循环。
  • 5、这里总结的是一次请求是怎么被分配执行的,而请求生成结果的核心还是在AysncCall的execute方法中的第147行,通过拦截器链生成返回结果。