Okhttp拦截器分为两大类,一类是应用拦截器,另一类是网络拦截器。
Application Interceptor
适用于在请求前统一加一些公共参数,例如在添加App的版本号,用户ID,手机版本号,运营商类型等参数。或者对响应体的数据进行json转化。
NetworkInterceptor
在这一层拦截器中可以获取到最终发送请求的request,也可以获取到真正发生网络请求后回来的response响应,从而修改对应的请求或者响应的数据。
两种拦截器的区别
Application Interceptor
- 不需要去关心发生的重定向和重试操作。因为它处于第一个拦截器,会获取到最终响应的response。
- 只会被调用一次,即使这个响应是从缓存中获取的。
- 只关注原始的请求,不去关心请求的资源是否发生了变化,我只关注最后的response结果而已。
- 因为是第一个被执行的拦截器,因此呢它有权决定是否要调用其他拦截,也就是Chain.proceed()方法是否要执行。
- 因为是第一个被执行的拦截器,因此它可以有多次调用Chain.proceed()方法,其实也就是相当于与重新请求的作用。
Network Interceptor
- 因为NetWorkInterceptor是排在第6个拦截器中,因此可以操作经过RetryAndFollowup进行失败重试或者重定向之后得到response。
- 对于从缓存获取的response则不会去触发NetworkInterceptor。因为响应会从CacheInterceptor返回。
- 观察数据在网络中的传输。
- 可以获得装在请求的连接。
例如自定义token拦截器(示例):
public class TokenInterceptor implements Interceptor {
public static int NEED_AUTH=401;
@Override
public Response intercept(Chain chain) throws IOException {
Request request=chain.request();
Response originalResponse=chain.proceed(request);
//401表示需要重新登陆或者token失效,大致的意思是
//当发起请求发现服务器返回token值过期的信息,这个时候就重新获取最新的Token值然后重新发起请求
if(originalResponse.code()==NEED_AUTH){
//TODO 获取新token,这里只给出思路
String newToken="newToken";
//然后保存token值
Request newRequest=request.newBuilder()
.header("token",newToken)//为请求重新添加Token值
.build();
originalResponse.body().close();
// RequestBody requestBody= new FormBody.Builder().build();
return chain.proceed(newRequest);
}
return originalResponse;
}
}
总结
不管是Application Interceptor还是NetWork Interceptor这两种拦截器都有一个共同的功能,那就是可以修改对应的请求或者是响应的数据。