安卓集中营Android干货安卓开发

okhttp之BridgeInterceptor

2018-04-09  本文已影响9人  黑猫警长_01

BridgeInterceptor是okhttp中的第二个拦截器,这个拦截器的主要作用就是在对用户的请求头部加了一些信息,然后在获取到的响应中也做了一些处理。而这些处理对用户是透明的,减少了客户请求的工作。下面来看一下对用户的请求做了哪些处理:

@Override public Response intercept(Chain chain) throws IOException {
Request userRequest = chain.request();
Request.Builder requestBuilder = userRequest.newBuilder();

RequestBody body = userRequest.body();
if (body != null) {
  //请求体类型
  MediaType contentType = body.contentType();
  if (contentType != null) {
    //封装头部的Content-Type
    requestBuilder.header("Content-Type", contentType.toString());
  }

  long contentLength = body.contentLength();
  //内容长度
  if (contentLength != -1) {
    requestBuilder.header("Content-Length", Long.toString(contentLength));
    requestBuilder.removeHeader("Transfer-Encoding");
  } else {
    requestBuilder.header("Transfer-Encoding", "chunked");
    requestBuilder.removeHeader("Content-Length");
  }
}

if (userRequest.header("Host") == null) {
  requestBuilder.header("Host", hostHeader(userRequest.url(), false));
}
//保持连接
if (userRequest.header("Connection") == null) {
  requestBuilder.header("Connection", "Keep-Alive");
}

// If we add an "Accept-Encoding: gzip" header field we're responsible for also decompressing
// the transfer stream.
boolean transparentGzip = false;
//gZip压缩
if (userRequest.header("Accept-Encoding") == null) {
  transparentGzip = true;
  requestBuilder.header("Accept-Encoding", "gzip");
}
//添加cookie(cookieJar用户可以自己配置,默认是一个空的实现,什么也不做)
List<Cookie> cookies = cookieJar.loadForRequest(userRequest.url());
if (!cookies.isEmpty()) {
  requestBuilder.header("Cookie", cookieHeader(cookies));
}

//用户代理
if (userRequest.header("User-Agent") == null) {
  requestBuilder.header("User-Agent", Version.userAgent());
}
//获取到response
Response networkResponse = chain.proceed(requestBuilder.build());

在上面的请求封装中,一共做了这么几件事:
1、封装头部的Content-Type
2、设置Content-Length或者Transfer-Encoding
3、如果host不存在,设置Host
4、如果头部没有设置Connection,设置Connection为Keep-Alive
5、如果没有设置压缩方式,设置gzip压缩
6、设置cookie,其中cookieJar是一个用来加载cookie和保存cookie的类,用户可以自己定义实现,默认是一个空的实现,什么都不做。
7、如果没有用户代理的话,设置用户代理对于请求的封装,一共就做了这么几件事,然后就是把请求交给其他的拦截器获取响应。

接下来看对response的处理

//处理接收到的cookie
HttpHeaders.receiveHeaders(cookieJar, userRequest.url(), networkResponse.headers());

Response.Builder responseBuilder = networkResponse.newBuilder()
    .request(userRequest);
//在发起请求时BridgeInterceptor给头部加了gzip压缩标记,并且响应头中有gzip压缩标记,进行解压
if (transparentGzip
    && "gzip".equalsIgnoreCase(networkResponse.header("Content-Encoding"))
    && HttpHeaders.hasBody(networkResponse)) {
  GzipSource responseBody = new GzipSource(networkResponse.body().source());
  //已经进行了解压,响应头中的Content-Encoding Content-Length已经无效了,所以移除
  Headers strippedHeaders = networkResponse.headers().newBuilder()
      .removeAll("Content-Encoding")
      .removeAll("Content-Length")
      .build();
  responseBuilder.headers(strippedHeaders);
  responseBuilder.body(new RealResponseBody(strippedHeaders, Okio.buffer(responseBody)));
}

return responseBuilder.build();}

这里面一共做了两件事:
1、获取到响应头中cookie,交给cookieJar进行保存
2、如果在本拦截器中为request加上了gzip压缩,并且响应的压缩方式也是gzip压缩,那么就把响应提就行解压。由于进行了解压,响应头中的Content-Encoding Content-Length就不正确了,所以移除掉这两个属性。

到这里这个拦截器就介绍完了,这应该是okhttp中最简单的一个拦截器了。

上一篇下一篇

猜你喜欢

热点阅读