• OkHttp3源码详解(三) 拦截器-RetryAndFollowUpInterceptor


    最大恢复追逐次数:

    private static final int MAX_FOLLOW_UPS = 20;

    处理的业务:

    1. 实例化StreamAllocation,初始化一个Socket连接对象,获取到输入/输出流()基于Okio
    2. 开启循环,执行下一个调用链(拦截器),等待返回结果(Response)
    3. 如果发生错误,判断是否继续请求,否:退出
    4. 检查响应是否符合要求,是:返回
    5. 关闭响应结果
    6. 判断是否达到最大限制数,是:退出
    7. 检查是否有相同连接,是:释放,重建连接
    8. 重复以上流程

    源码

     1 @Override 
     2 public Response intercept(Chain chain) throws IOException {
     3   // 
     4   Request request = chain.request();
     5  // 1. 初始化一个socket连接对象
     6   streamAllocation = new StreamAllocation(
     7       client.connectionPool(), createAddress(request.url()), callStackTrace);
     8 
     9   int followUpCount = 0;
    10   Response priorResponse = null;
    11   while (true) {
    12      // 
    13     if (canceled) {
    14       streamAllocation.release();
    15       throw new IOException("Canceled");
    16     }
    17 
    18     Response response = null;
    19     boolean releaseConnection = true;
    20     try {
    21        // 2. 执行下一个拦截器,即BridgeInterceptor
    22       response = ((RealInterceptorChain) chain).proceed(request, streamAllocation, null, null);
    23       releaseConnection = false;
    24     } catch (RouteException e) {
    25       // The attempt to connect via a route failed. The request will not have been sent.
    26        // 3. 如果有异常,判断是否要恢复
    27       if (!recover(e.getLastConnectException(), false, request)) {
    28         throw e.getLastConnectException();
    29       }
    30       releaseConnection = false;
    31       continue;
    32     } catch (IOException e) {
    33       // An attempt to communicate with a server failed. The request may have been sent.
    34       boolean requestSendStarted = !(e instanceof ConnectionShutdownException);
    35       if (!recover(e, requestSendStarted, request)) throw e;
    36       releaseConnection = false;
    37       continue;
    38     } finally {
    39       // We're throwing an unchecked exception. Release any resources.
    40       if (releaseConnection) {
    41         streamAllocation.streamFailed(null);
    42         streamAllocation.release();
    43       }
    44     }
    45 
    46     // Attach the prior response if it exists. Such responses never have a body.
    47     if (priorResponse != null) {
    48       response = response.newBuilder()
    49           .priorResponse(priorResponse.newBuilder()
    50                   .body(null)
    51                   .build())
    52           .build();
    53     }
    54  // 4. 检查是否符合要求
    55     Request followUp = followUpRequest(response);
    56 
    57     if (followUp == null) {
    58       if (!forWebSocket) {
    59         streamAllocation.release();
    60       }
    61       // 返回结果
    62       return response;
    63     }
    64  // 5. 不符合,关闭响应流
    65     closeQuietly(response.body());
    66  // 6. 是否超过最大限制
    67     if (++followUpCount > MAX_FOLLOW_UPS) {
    68       streamAllocation.release();
    69       throw new ProtocolException("Too many follow-up requests: " + followUpCount);
    70     }
    71 
    72     if (followUp.body() instanceof UnrepeatableRequestBody) {
    73       streamAllocation.release();
    74       throw new HttpRetryException("Cannot retry streamed HTTP body", response.code());
    75     }
    76  // 7. 是否有相同的连接
    77     if (!sameConnection(response, followUp.url())) {
    78       streamAllocation.release();
    79       streamAllocation = new StreamAllocation(
    80           client.connectionPool(), createAddress(followUp.url()), callStackTrace);
    81     } else if (streamAllocation.codec() != null) {
    82       throw new IllegalStateException("Closing the body of " + response
    83           + " didn't close its backing stream. Bad interceptor?");
    84     }
    85 
    86     request = followUp;
    87     priorResponse = response;
    88   }
    89 }

    初始化连接对象

    streamAllocation = new StreamAllocation(
            client.connectionPool(), createAddress(request.url()), callStackTrace);

    注意:此处还没有真正的去建立连接,只是初始化一个连接对象

    继续下一个拦截器

    上面一步初始化好后,将继续执行下一个连接器BridgeInterceptor,

    // 这里有个很重的信息,即会将初始化好的连接对象传递给下一个拦截器,也是贯穿整个请求的连击对象,
    // 上文我们说过,在拦截器执行过程中,RealInterceptorChain的几个属性字段会一步一步赋值
    response = ((RealInterceptorChain) chain).proceed(request, streamAllocation, null, null);

    抛出异常
    如果抛出异常,将判断是否能够继续连接,以下情况不在,重试:

     1 /**
     2 * 不在继续连接的情况:
     3 * 1. 应用层配置不在连接,默认为true
     4 * 2. 请求Request出错不能继续使用
     5 * 3. 是否可以恢复的
     6 *   3.1、协议错误(ProtocolException)
     7     3.2、中断异常(InterruptedIOException)
     8     3.3、SSL握手错误(SSLHandshakeException && CertificateException)
     9     3.4、certificate pinning错误(SSLPeerUnverifiedException)
    10 * 4. 没用更多线路可供选择
    11 */
    12 private boolean recover(IOException e, boolean requestSendStarted, Request userRequest) {
    13   streamAllocation.streamFailed(e);
    14   // 1. 应用层配置不在连接,默认为true
    15   // The application layer has forbidden retries.
    16   if (!client.retryOnConnectionFailure()) return false;
    17 
    18   // 2. 请求Request出错不能继续使用
    19   // We can't send the request body again.
    20   if (requestSendStarted && userRequest.body() instanceof UnrepeatableRequestBody) return false;
    21 
    22   //  是否可以恢复的
    23   // This exception is fatal.
    24   if (!isRecoverable(e, requestSendStarted)) return false;
    25 
    26   // 4. 没用更多线路可供选择
    27   // No more routes to attempt.
    28   if (!streamAllocation.hasMoreRoutes()) return false;
    29 
    30   // For failure recovery, use the same route selector with a new connection.
    31   return true;
    32 }

     http://lowett.com/categories/Android/Okhttp3/

  • 相关阅读:
    Memcached:高性能的分布式内存缓存服务器
    MySQL数据库Query的优化
    MySQL数据库的锁定机制及优化
    系统架构及实现对性能的影响(一)
    Mysql数据库的基本结构和存储引擎简介
    Spring事务管理的回滚
    穷举算法实例
    在写完全二叉树的构建及遍历
    Inotify
    Rsync扩展
  • 原文地址:https://www.cnblogs.com/ganchuanpu/p/6880555.html
Copyright © 2020-2023  润新知