深入了解OkHttp3之Interceptors

OKHttp官网上面介绍:拦截器是一种强大的机制,可以监视,重写和重试调用。这里我们主要对拦截器的网络请求头和数据请求的封装。

网络请求头拦截器

在Android应用中,我们通常需要获取用户手机的IMEI值和其他一些常用的参数,如果把他们封装到请求头里面的话会非常的方便。

既然是自定义拦截器,我们首先肯定要继承OKHttp的拦截器,把这个自定义类设置成 final 类型的防止其他类继承。

这里需要在父类的intercept(Chain chain)方法中进行操作。在方法里面得Request.Builder对象然后添加请求头。

创建一个BaseParam 对象,对请求头的参数进行封装。

public final class HeaderInterceptorTest implements Interceptor {

  /**
   * 请求头参数基础参数
   */
  private static final String HEADER_BASE_PARAM = "baseParam";
  private static String BASE_PARAM;

  @Override
  public Response intercept(Chain chain) throws IOException {

    Request.Builder builder = chain.request().newBuilder();

    builder.addHeader(HEADER_BASE_PARAM, BASE_PARAM);
    return chain.proceed(builder.build());
  }

  public static void initBaseParam(Context context) {
    BaseParam baseParam = new BaseParam();
    String imei;
    String imsi;
    try {
      imei = DeviceInfoUtil.getIMEI(context);
    } catch (Exception e) {
      imei = null;
    }
    if (null == imei || imei.equals("")) {
      imei = DeviceInfoUtil.getDeviceID(context);
      imsi = DeviceInfoUtil.getDeviceID(context);
    } else {
      imei = DeviceInfoUtil.getIMEI(context);
      imsi = DeviceInfoUtil.getIMSI(context);
    }
    baseParam.setImei(imei);
    baseParam.setImsi(imsi);
    baseParam.setMac(DeviceInfoUtil.getWifiMAC(context));
    baseParam.setVersion(DeviceInfoUtil.getVersion(context));
    baseParam.setModel(DeviceInfoUtil.getModel());
    baseParam.setBrand(DeviceInfoUtil.getBrand());
    BASE_PARAM = new Gson().toJson(baseParam).toString();
  }

  private static class BaseParam {
    /**
     * imei :
     * imsi :
     * mac :
     * version :
     * model :
     * brand :
     * city :
     */
    private String imei;
    private String imsi;
    private String mac;
    private String version;
    private String model;
    private String brand;
    private String city;

    public String getImei() {
      return imei;
    }

    public void setImei(String imei) {
      this.imei = imei;
    }

    public String getImsi() {
      return imsi;
    }

    public void setImsi(String imsi) {
      this.imsi = imsi;
    }

    public String getMac() {
      return mac;
    }

    public void setMac(String mac) {
      this.mac = mac;
    }

    public String getVersion() {
      return version;
    }

    public void setVersion(String version) {
      this.version = version;
    }

    public String getModel() {
      return model;
    }

    public void setModel(String model) {
      this.model = model;
    }

    public String getBrand() {
      return brand;
    }

    public void setBrand(String brand) {
      this.brand = brand;
    }

    public String getCity() {
      return city;
    }

    public void setCity(String city) {
      this.city = city;
    }
  }
}

数据请求拦截器

在进行数据请求的时候,一般都会定义一种固定的请求格式。在这里我们直接通过拦截器,将这个固定的格式定义好,这样使用起来可以统一管理。具体的使用方法其实和上面一样。

主要注意的是先得到,请求时设置的请求体,然后让请求的数据封装到和后端设置的请求格式里面。

public class HttpDataPackInterceptorTest implements Interceptor {

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

    Request.Builder requestBuilder = chain.request().newBuilder();
    //请求数据处理
    if (request.url().toString().contains("uploads")) {
      //特殊接口的参数不需要处理的
    } else {
      //需要处理请求参数的
      if (request.body() instanceof FormBody) {
        FormBody.Builder newFormBody = new FormBody.Builder();
        //得到请求时设置的请求体
        FormBody oldFormBody = (FormBody) request.body();
        Buffer buffer = new Buffer();
        oldFormBody.writeTo(buffer);
        String postParams = JsonUtils.getJsonStrFromPostParams(buffer.readString(Charset.forName("UTF-8")));
        String data = URLDecoder.decode(postParams, "UTF-8");
        if (TextUtils.isEmpty(data)) {
          newFormBody.add("data", "");
        } else {
          //将请求的数据封装
          newFormBody.add("data", data);
        }
        requestBuilder.method(request.method(), newFormBody.build());
      }
    }
    return chain.proceed(requestBuilder.build());
  }
}

代码中调用

在创建OkHttpClient 对象的时候,调用 addInterceptor()方法 添加俩个拦截器。

private OkHttpClient client;
  @Override
  protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_okhttp);
    HeaderInterceptorTest.initBaseParam(this);
    client= new OkHttpClient()
        .newBuilder()
        .addInterceptor(new HeaderInterceptorTest())
        .addInterceptor(new HttpDataPackInterceptorTest())
        .connectTimeout(60, TimeUnit.SECONDS)
        .writeTimeout(60, TimeUnit.SECONDS)
        .readTimeout(60, TimeUnit.SECONDS).build();
    okhttpAsyncPost();
  }

  private void okhttpAsyncPost(){
    RequestBody formBody = new FormBody.Builder()
        .add("page", "1")
        .add("count", "2")
        .add("type","video")
        .build();
    Request request = new Request.Builder()
        .url("https://api.apiopen.top/getJoke")
        .post(formBody)
        .build();

    client.newCall(request).enqueue(new Callback() {
      @Override
      public void onFailure(Call call, IOException e) {

      }

      @Override
      public void onResponse(Call call, Response response) throws IOException {
        Log.e("error","结果"+response.body().string());
        Log.e("error","方法"+response.request().toString());
        Log.e("error","请求头"+response.request().headers().toString());
      }
    });
  }

打印的日志

结果{"status":200,"msg":"OK","data":"{\"count\":\"2\",\"page\":\"1\",\"type\":\"video\"}"}
方法Request{method=POST, url=https://www.26uuun.com/list, tags={}}
请求头baseParam: {"brand":"Xiaomi","city":"\u4e1c\u4eac","imei":"9fc70b16bf169075f556e6d67be9ef1a","imsi":"9fc70b16bf169075f556e6d67be9ef1a","mac":"02:00:00:00:00:00","model":"Mi Note 2","version":"1.0"}

OK,OKHttp3中Interceptors配置完成,小伙伴们,可以设置自己不同需求的其他的拦截器添加到请求中。

最近我会写很多关于Android常用控件的使用,里面都是一些很有用的知识,如果你感觉有用,请给我一个star,谢谢。 代码实例

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 深入了解OkHttp3之Interceptors

    OKHttp官网上面介绍:拦截器是一种强大的机制,可以监视,重写和重试调用.这里我们主要对拦截器的网络请求头和数据请求的封装. 网络请求头拦截器 在Android应用中,我们通常需要获取用户手机的IMEI值和其他一些常用的参数,如果把他们封装到请求头里面的话会非常的方便. 既然是自定义拦截器,我们首先肯定要继承OKHttp的拦截器,把这个自定义类设置成 final 类型的防止其他类继承. 这里需要在父类的intercept(Chain chain)方法中进行操作.在方法里面得Request.Bu

  • 解析Android框架之OkHttp3源码

    OkHttp流程图 OkHttp基本使用 gradle依赖 implementation 'com.squareup.okhttp3:okhttp:3.11.0' implementation 'com.squareup.okio:okio:1.15.0' /** *这里拿get请求来 * 异步的get请求 */ public void okhttpAsyn() { //设置超时的时间 OkHttpClient.Builder builder = new OkHttpClient.Builder

  • angular中的http拦截器Interceptors的实现

    在angularJs中增加了一个对全局的http请求统一做出处理的api--interceptors Interceptors 有两个处理时机,分别是: 其它程序代码执行 HTTP 请求之后,在实际从浏览器发出请求之前,即处理请求 得到请求的响应之后,在交给其它程序代码处理之前,即处理请求的响应 其引用场景包括 全局处理错误 统一进行身份验证一类的处理 对所有发出去的请求进行预处理 对所有收到的响应进行预处理等等 使用实例如下: commonService.config(['$httpProvi

  • Angular中的interceptors拦截器

    废话不多说了,直接给大家贴代码了,具体代码如下所述: <!DOCTYPE html> <html ng-app="nickApp"> <head> <meta charset="UTF-8"> <meta name="viewport" content="initial-scale=1, maximum-scale=1, user-scalable=no, width=device-

  • 详解AngularJS用Interceptors来统一处理HTTP请求和响应

    Web 开发中,除了数据操作之外,最频繁的就是发起和处理各种 HTTP 请求了,加上 HTTP 请求又是异步的,如果在每个请求中来单独捕获各种常规错误,处理各类自定义错误,那将会有大量的功能类似的代码,或者使用丑陋的方法在每个请求中调用某几个自定义的函数来处理.这两种方法基本都不是靠谱之选.好在 AngularJS 提供了 Interceptors --拦截战斗机--来对应用内所有的 XHR 请求进行统一处理. 主要功能 Interceptors 有两个处理时机,分别是: 其它程序代码执行 HT

  • Android中Okhttp3实现上传多张图片同时传递参数

    之前上传图片都是直接将图片转化为io流传给服务器,没有用框架传图片. 最近做项目,打算换个方法上传图片. Android发展到现在,Okhttp显得越来越重要,所以,这次我选择用Okhttp上传图片. Okhttp目前已经更新到Okhttp3版本了,用法跟之前相比,也有一些差别.在网上找了很多资料, 并和java后台同事反复调试,终于成功上传多张图片,同时传递一些键值对参数. 以下是我对该过程的封装: private static final MediaType MEDIA_TYPE_PNG =

  • Okhttp3实现爬取验证码及获取Cookie的示例

    目前正在做毕业设计,一个关于校园服务的app,我会抽取已完成的相关代码写到文章里.一是为了造福这个曾经帮助过我的社区,二是写文章的同时更能巩固相关知识的记忆. 一.前言 在爬取教务系统的过程中,验证码的获取是非常重要的:在生成验证码的图片的同时,服务器会返回一个cookie,在接下来的所有请求都需要带上这个cookie. 二.使用前准备 2.1 简介 Okhttp是一个快速.高效的网络请求库.详情可以查阅官方文档 2.2 Android Studio配置gradle compile'com.sq

  • vue拦截器Vue.http.interceptors.push使用详解

    刚开始学vue,github上down了一个开源项目,看源代码的时候看到了这个地方: /** * @export * @param {any} request * @param {any} next * @returns */ import store from './vuex/store' // 全局错误处理,全局loading import { setLoading, setTip } from './vuex/actions/doc_actions' export default func

  • 详解okhttp3 请求头不能为中文的坑

    对源码有一定的了解,以为基本可以驾驭了,谁知道,坑来了. 问题描述: 上线接口的某一天,有用户报障,说登录不上,而且陆续有报障. 通过log 发现登录不上,都有一个类似的报错: "java.lang.IllegalArgumentException: Unexpected char 0x514d at 4 in wifiSsid value: "360免费WiFi-DP" at " 什么?怎么和wifiSsid有关系了?后来查了一下代码,发现确实在登录接口上通过请求

  • Android中okhttp3.4.1+retrofit2.1.0实现离线缓存

    关于Retrofit+OkHttp的强大这里就不多说了,还没了解的同学可以自行去百度.这篇文章主要讲如何利用Retrofit+OkHttp来实现一个较为简单的缓存策略: 即有网环境下我们请求数据时,如果没有缓存或者缓存过期了,就去服务器拿数据,并且将新缓存保存下来,如果有缓存而且没有过期,则直接使用缓存.无网环境下我们请求数据时,缓存没过期则直接使用缓存,缓存过期了则无法使用,需要重新联网获取服务器数据. 缓存处理还是很有必要的,它有效的减少服务器负荷,降低延迟提升用户体验,同时也方便用户即使在

随机推荐