okhttp 过滤器 Interceptors 中ResonseBody 只能读取一次
由于公司的项目要不定时的去刷新 access token 这就需要在token 过期之后自动的去获取一次,当时首先想到就是使用过滤器,Interceptors,但是okhttp 的返回Resonse有一个坑 就是Resonse.body().string()或者是 charStream()方法只能读一次,这是为了节省内存的开销,但是对于初次使用的okhttp的菜鸟的我来说这就个大坑啊,事实上他的Interceptors 中返回Resonse对象和最终返回的Resonse对象是同一个对象,我当时就懵逼了!Resonse 只能读一次!而我们过滤器的初衷是为了判断token是否过期,而token的过期判断又在返回内容中,在这里直接读取了,回调中就读取不到了,没办法就只好苦逼的去读取源码了,生成一个ResonseBody 对象的clone.以此记录,避免将来那一天自己在忘记了.
public class ResponseInterceptor implements Interceptor {
@Override
public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
Response response = chain.proceed(request);
ResponseBody copyBody = cloneResponseBody(response);
if (copyBody != null) {
//do something
//这里获取到的 ResponseBody 是一个新的对象不会对原来的ResponseBody产生影响
String string = copyBody.string();
LOG.i(string);
} else {
return response;
}
return response;
}
/**
* @param response 当前请求的返回对象
* @return 当前请求的返回对象的Body 的克隆对象
*/
private ResponseBody cloneResponseBody(Response response) {
/**
* 这里的方法会对 ResponseBody的 source 进行克隆并返回指定的长度!
* 我们需哟的是是一份完全的克隆体 所以传入的就是原ResponseBody的 长度
*/
ResponseBody result;
try {
ResponseBody body = response.body();
BufferedSource source = body.source();
source.request(Long.MAX_VALUE);
Buffer buffer = source.buffer();
Buffer copyBuffer = buffer.clone();
/***
* 也可以直接生成String 返回
*/
// String json = copyBuffer.readString(charset(body));
result = new RealResponseBody(response.headers(), copyBuffer);
// result = ResponseBody.create(body.contentType(), copyBuffer.size(), copyBuffer);
} catch (IOException e) {
result = null;
}
return result;
}
/**
* @param body ResponseBody 对象
* @return ResponseBody的字符集, 默认utf-8
*/
private Charset charset(ResponseBody body) {
MediaType contentType = body.contentType();
return contentType != null ? contentType.charset(Util.UTF_8) : Util.UTF_8;
}
}