Okhttp的两种请求方式
同步请求
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()
.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()
.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()
.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);
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(); 至此异步任务也分析完毕
|