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();
至此异步任务也分析完毕