OkHttp源码彻底解析(三)OkHttp3.0拦截器原理——责任链模式

 

本系列文章:

OkHttp源码彻底解析(一)OkHttp请求流程

OkHttp源码彻底解析(二)OkHttp架构及API源码

OkHttp源码彻底解析(三)OkHttp3.0拦截器原理——责任链模式

OkHttp源码彻底解析(四)OkHttp拦截器的作用

OkHttp源码彻底解析(五)OkHttp连接池


目录

什么是拦截器

拦截器中的类

拦截器中的源码

拦截器源码的逻辑流程


什么是拦截器

拦截器是OkHttp中提供一种强大机制,它可以实现网络监听、请求以及响应重写、请求失败重试等功能。

Okhttp3.0中的拦截器有点像安卓里面的触控反馈的Interceptor。既一个网络请求,按一定的顺序,经由多个拦截器进行处理,该拦截器可以决定自己处理并且返回我的结果,也可以选择向下继续传递,让后面的拦截器处理返回它的结果。这个设计模式叫做责任链模式。

与Android中的触控反馈interceptor的设计略有不同的是,后者通过返回true 或者 false 来决定是否已经拦截。而OkHttp这里的拦截器通过函数调用的方式,讲参数传递给后面的拦截器的方式进行传递。这样做的好处是拦截器的逻辑比较灵活,可以在后面的拦截器处理完并返回结果后仍然执行自己的逻辑;缺点是逻辑没有前者清晰。

拦截器中的类

在拦截器这种结构中,我们只要能掌握好以下两个角色就可以了

    Interceptor                这个我们的拦截器接口,所有拦截器都要实现它

    Chain                         这个是串联起拦截器流程的链

下面来看看这两个类中的逻辑:

1.Chain中持有所有要调用的Interceptor 的列表集合incps

Chain中的proceed方法:调用调用每个Interceptor 的intercept方法 会把Chain自己和去掉第一个Interceptor 的incpts(已经调用完的Interceptor 后面就不调用)作为参数传入

2.Interceptor 中

intercept方法:处理自身负责的逻辑,该方法在Chain中被调用并传入Chain本身,在执行了自身的逻辑之后,调用传入的Chain的proceed

总结就是,Chain中的procced与Interceptor 中的intercept相互循环调用,而procced中每次Chain中的Interceptor 都会往后移动一位,所以这个循环当所以Interceptor 执行过后就会终止。

拦截器中的源码

那我们来看看他们通过源码来看看他们内部是如何做到有序地串联的。

拦截器接口的源码:

public interface Interceptor {
 Response intercept(Chain chain) throws IOException;

 interface Chain {
   Request request();

   Response proceed(Request request) throws IOException;

   Connection connection();
 }
}

其中的Chain就是我们是用来传递的链。这里的传递逻辑伪代码如下:
代码的最外层逻辑(下面这个不只是chain的,是整个的伪代码)

Request request = new Request(){};
 
 
 Arrlist<Interceptor> incpts = new Arrlist();
 Interceptor icpt0 = new Interceptor(){ XXX };
 Interceptor icpt1 = new Interceptor(){ XXX };
 Interceptor icpt2 = new Interceptor(){ XXX };
 ...
 incpts.add(icpt0);
 incpts.add(icpt1);
 incpts.add(icpt2);
 
 Interceptor.Chain chain  = new MyChain(incpts);
 chain.proceed(request);

下面就是最关键的源码部分,基本上所以的责任链Chain都是按着这个模板来的

封装的Chain的内部逻辑

public class MyChain implement Interceptor.Chain{
    Arrlist<Interceptor> incpts;
    int index = 0;
    
    public MyChain(Arrlist<Interceptor> incpts){
        this(incpts, 0);
    }
    
    public MyChain(Arrlist<Interceptor> incpts, int index){
        this.incpts = incpts;
        this.index =index;
    }
    
    public void setInterceptors(Arrlist<Interceptor> incpts ){
        this.incpts = incpts;
    }
 
    @override
    Response proceed(Request request) throws IOException{
            Response response = null;
            ...
            //取出第一个interceptor来处理
            Interceptor incpt = incpts.get(index);
            //生成下一个Chain,index标识当前Interceptor的位置。
            Interceptor.Chain nextChain = new MyChain(incpts,index+1);
            response =  incpt.intercept(nextChain);
            ...
            return response;
    }
  }

而各个Interceptor源码中的实现:

public class MyInterceptor implement Intercetpor{
    @Override 
    public Response intercept(Chain chain) throws IOException {
        Request request = chain.request();
        //前置拦截逻辑
        ...
        Response response = chain.proceed(request);//传递Interceptor
        //后置拦截逻辑
        ...
        return response;
    }
}

在这个链中,最后的一个Interceptor一般用作生成最后的Response操作,它不会再继续传递给下一个。

拦截器源码的逻辑流程

上面就是整个逻辑的源码了,虽然不多,但是逻辑有点绕(莫慌,看了下面你就会懂了)

这里为了让调用顺序更直观些,直接把源码的调用关系画出

这是Interceptor和Chain的源码以及方法的调用关系,是不是有点像递归函数的感觉。不过两者还是有区别的,递归函数是不断调用自身,而这种拦截器的逻辑是两个类(或者说是类内部的方法)不断相互调用,以某个条件作为终结。

诸如此类的逻辑还有   自定义控件的点击事件拦截,ClassLoader的双亲委派机制,其他的这里就不做展开

关于其中的顺序,用流程图来看更直观  (其中,所有Chain都是同一个,括号内编号表示第几次调用proceed方法)

说了这么多都是源码,这里附上OkHttp中实际各个拦截器的调用顺序,方便大家比照一下

       以上就是拦截器的整个流程,关于拦截器的具体运用,我将在另一篇文章中详细阐述

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
OkHttp是一个用于处理HTTP请求的开源Java库。它提供了一个拦截器机制,可以在发送请求和接收响应之前对它们进行修改和处理。以下是关于OkHttp拦截器的一些介绍和示例: 1. OkHttp拦截器是一个接口,它有一个方法intercept(Chain chain),该方法接收一个Chain对象作为参数,该对象表示当前的拦截器链。 2. 拦截器链是按照添加顺序执行的,每个拦截器都可以选择将请求传递给下一个拦截器或者直接返回响应。 3. 拦截器可以在请求和响应中添加、修改或删除头信息,也可以重试请求或者记录请求和响应的日志等。 以下是一个简单的OkHttp拦截器示例,它会在请求头中添加一个自定义的User-Agent信息: ```java public class UserAgentInterceptor implements Interceptor { private static final String USER_AGENT_HEADER = "User-Agent"; private final String userAgent; public UserAgentInterceptor(String userAgent) { this.userAgent = userAgent; } @Override public Response intercept(Chain chain) throws IOException { Request request = chain.request(); Request newRequest = request.newBuilder() .header(USER_AGENT_HEADER, userAgent) .build(); return chain.proceed(newRequest); } } ``` 在上面的示例中,我们创建了一个名为UserAgentInterceptor的拦截器,它接收一个User-Agent字符串作为参数。在intercept方法中,我们首先获取当前的请求对象,然后使用Request.Builder添加一个自定义的User-Agent头信息,最后使用chain.proceed方法将请求传递给下一个拦截器或者返回响应。 以下是一个使用上面定义的拦截器的示例: ```java OkHttpClient client = new OkHttpClient.Builder() .addInterceptor(new UserAgentInterceptor("MyApp/1.0")) .build(); ``` 在上面的示例中,我们创建了一个OkHttpClient对象,并使用addInterceptor方法添加了一个UserAgentInterceptor拦截器。这样,在发送请求时,OkHttp会自动调用我们定义的拦截器,并在请求头中添加一个自定义的User-Agent信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值