OkHttp源码解析

Okhttp的两种请求方式

同步请求

 /**
         * 在进行网络请求的时候不要忘记在AndroidManifest.xml中添加网络权限
         * <uses-permission android:name="android.permission.INTERNET"/>
         * 由于android不允许在UI线程中进行同步网络请求,所以我们用一个线程发起网络请求
         */
        new Thread(new Runnable() {
            @Override
            public void run() {
                OkHttpClient client = new OkHttpClient.Builder().build();
                Request request = new Request.Builder()
                        .url("https://www.baidu.com")
                        .get()//.post(RequestBody)post请求
                        .build();
                try {
                    Response response = client.newCall(request).execute();
                    LogUtil.e("我是同步请求的内容 " + response.body().string());
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }).start();

异步请求

OkHttpClient client = new OkHttpClient.Builder().build();
        Request request = new Request.Builder()
                .url("https://www.baidu.com")
                .get()//.post(RequestBody)post请求
                .build();
        try {
            client.newCall(request).call.enqueue(new Callback() {
                @Override
                public void onFailure(Call call, IOException e) {
                    LogUtil.e("我是异步请求的错误 " + e.getMessage());
                }

                @Override
                public void onResponse(Call call, Response response) throws IOException {
                    LogUtil.e("我是异步请求的内容 " + response.body().string());
                }
            });

        } catch (Exception e) {
            e.printStackTrace();
        }

同步请求分析源码

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

在这里插入图片描述

OkHttpClient 通过构建者模式进行创建客户端,在OkHttpClient可以设置以上配置,如果我们没有配置则使用OkHttp中默认的参数

Request request = new Request.Builder()
                        .url("https://www.baidu.com")
                        .get()//.post(RequestBody)post请求
                        .build();

在这里插入图片描述
Request 里面就是配置我们的请求模式例如我们的请求头,URL,请求方法(get()或者是post()请求);

 Response response = client.newCall(request).execute();//以同步的方式请求网络

client.newCall(request) 这句代码是调用OkhttpClient里面的newCall方法,在这里面他创建了一个RealCall对象并且返回给我们
在这里插入图片描述
client.newCall(request).execute()等价于RealCall.execute() 同步方式请求网络,我们看一下他的源码在这里插入图片描述
在79行执行的函数调用的是Dispatch类里面的excuted(),很简单只是把网络请求任务添加到runningSyncCalls队列当中,方便后续做统计
在这里插入图片描述
82行执行的函数是真正的网络请求同时也是我们5大拦截器质性的地方
在这里插入图片描述
在84行会返回我们的Response,最后90行无论请求成功还是失败都会把本次任务在runningSyncCalls队列当中移除

到此我们的同步请求分析完毕

同步请求分析源码

在异步请求我们直接来分发器分发任务的代码RealCall.enqueue()方法中

 @Override public void enqueue(Callback responseCallback) {
    //
    synchronized (this) {
      if (executed) throw new IllegalStateException("Already Executed");
      executed = true;
    }
    //监听事件
    captureCallStackTrace();
    eventListener.callStart(this);

    //执行分发器中的enqueue()并且把任务封装成 AsyncCall AsyncCall集成NamedRunable
    client.dispatcher().enqueue(new AsyncCall(responseCallback));
  }

我们先来看AsyncCall类代码,其中execute()是最关键的认为会在namedRunnable中调用
在这里插入图片描述
在这里插入图片描述

client.dispatcher().enqueue(new AsyncCall(responseCallback))等价值与Dispatch.enqueue(new AsyncCall(responseCallback)),所以我们直接看Dispatch.enqueue()方法里面的源码
在这里插入图片描述

在上面的图片中的第151行调度使用线程池进行执行AsyncCall.execute()方法,我们先来看线程池是怎么初始化的
在这里插入图片描述
再来看我们的AsyncCall.execute()方法,这里面是网络请求的关键
在这里插入图片描述
getResponseWithInterceptorChain()方法和同步网络请求是一样的这里不在赘述,我们看在网络请求之后是怎么调用readyAsyncCalls队列的任务
在这里插入图片描述
我们继续看239行的执行的函数promoteCalls();
在这里插入图片描述
至此异步任务也分析完毕


版权声明:本文为qq_41740162原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。