1. 框架功能简介:暂时只有get、post两个请求

2. 请求的主要流程和区别:

2.1 get请求:

(1)创建请求客户的 OkHttpClient对象

(2)创建请求构建器 Request.Bulder builder = new Request.Builder().url(...). build();

(3)创建请求对象:Request request = builder.build();

(3)创建请求任务 Call call

(4)发起异步请求:call.enqueue(new CallBack(){...});

2.2 post请求

(1)创建请求客户的 OkHttpClient对象

(2)创建请求构建器 Request.Builder builder = new Request.Builder().url(...). build();

(3)创建表单实体构建器:FormBody.Builder mFormBuilder

(4)创建表单实体对象:FormBody body

(5)创建请求对象:Request request = builder.post(body);

(6)创建请求任务:Call call

(7)发起异步任务:call.enqueue(new CallBack(){...});

2.3 两者区别

请求构建器 builder 与请求对象request之间,post需要创建请求实体,并在请求实体中添加参数——封装的框架需要考虑两者区别,处理好复用代码和分别处理

3. 代码

/**

*OkHttp的二次封装

*get请求、post请求

* 客户端异步发起请求,(url(), targetClass(), 调用接口等)——OkUtils通过Handler返回结果

*/

public class OkUtils<T>{

  private static final String UTF_8 = "utf-8";

  private static final int RESULT_SUCCESS = 0;

  private static final int RESULT_ERROR =1;

  private StringBuilder mUrl;

  private OkHttpClient mOkHttpClient; // 单例模式

  private Handler mHandler;

  private OnCompleteListener<T> mListener;

  // 地址栏请求

  private StringBuilder mUrl;

  // 泛型,目标解析类

  private Class mClaz;

  // post表单body构造器

  private FromBody.Builder mFormBuilder;

  

  // 对外提供接口:客户端向服务端发送请求、处理返回结果的接口

  public interface OnCompleteListener<T>{

    void onSuccess(T t);

    void onError(String error);

  }

  // 构造方法,单例模式初始化mOkHttpClient

  pubic OkUtils(Context context){// 注意参数要有

    if(mOkHttpClient == null){

      synchronized(OkUtils.class){

        if(mOkHttpClient == null){

          mOkHttpClient = new OkHttpClient();

        }

      }

    }

    initHandler(context);

  }

  private void initHandler(Context context){

    mHandler = new Handler(context.getMainLooper){

      @override

      public void handleMessage(Message msg){

        super.handleMessage(msg);

        switch(msg.what){

          case RESULT_SUCCESS:

            if(mListener != null && msg.obj != null){

              T t = (T) mst.obj;        

              mListener.onSuccess(t);

            }       

            break;

          case RESULT_ERROR:

            if(mListener != null && msg.obj != null){

              mListener.onError(msg.obj.toString());

            }         

            break;

        }

      }

    };

  }

  // 地址栏请求

  public OkUtils<T> url(StringBuilder url){

    mUrl = new StringBuilder(url);

    return this;

  }

  // 创建post表单body构建器

  public OkUtils<T> post(){

    mFormBuilder = new FormBody.Builder();

    return this;

  }

  // 目标解析类

  public OkUtils<T> targetClass(Class claz){

    mClaz = claz;

  }

  // 添加参数

  public OkUtils<T> addParam(String key, String value){

    if(mFormBuilder != null){ // post请求的参数添加方式

      mFormBuilder.add( key, URLEncoder.encode(value, UTF_8));

    }else{// get请求的参数添加方式

      if(mUrl.indexOf("?") == -1){

        mUrl.apend("?");

      }else{

        mUrl.apend("&");

      }

      mUrl.append(key).append("=").append(URLEncoder.encode(value, UTF_8));  // 注意此处需要抛异常,已省略。编辑代码时会遇到。

    }

    return this;

  }

  public void execute(OnCompleteListener listener){

    mListener = listener;

    // 校验 mUrl,参数,mClaz是否符合发送请求的要求

    if( mUrl == null){

      Message msg = Message.obtain();

      msg.what = RESULT_ERROR;

      msg.obj = "url不能为空";

      mHandler.sendEmptyMessage(msg);

      return;

    }

    if( mUrl.indexOf("?" == -1){

      Message msg = Message.obtain();

      msg.what = RESULT_ERROR;

      msg.obj = "请设置请求参数";

      mHandler.sendEmptyMessage(msg);

    }

    if( mClaz == null){

      Message msg = Message.obtain();

      msg.what = RESULT_ERROR;

      msg.obj = "解析目标类claz不能为空";

      mHandler.sendEmptyMessage(msg);

      return;

    }

    // 创建请求构建器

    Request.Builder builder = new Request.Builder().url(mUrl.toString);

    if(mFormBuilder != null){ // post请求,创建表单实体

      FormBody body = mFormBuilder.build();

      builder.post(body);

    }

    Request request = builder.build();

    Call call = mOkHttpClient.newCall(request);

    call.enqueue(new CallBack(){

      @override

      public void onFailure(Call call, IOException e){

        Message msg = Message.obtain();

        msg.what = RESULT_ERROR;

        msg.obj = e.toString();

        mHandler.sendMessage(msg);

      }

      @override

      public void onResponse(Call call, Response response) throws IOExcepton{

        String json = response.body().string();

        T t = (T)ResultUtils.getResultFromJson(json, mClaz); // ResultUtils 是一个工具类,解析json数据,如果需要请留言

        Message msg = Message.obtain();

        msg.what = RESULT_OK;

        msg.obj = t;

        mHandler.sendMessage(msg);

      }

    });

  }

  public static void release(){

    if( mOkHttpClient != null) {

      mOkHttpClient.dispatcher().cancel();

      mOkHttpClient == null;

    }

  }

}

到此,一个基本的OkHttp就封装好了,其他的诸如上传文件、下载文件等功能,下次整理。

二次封装OKHttp网络框架(1)的更多相关文章

  1. Android OKHttp网络框架

    好久没逛简书了.这周公司的项目也已经愉快的迭代了新版本,对于之前一直存留的东西一直没怎么梳理,今天想说说这两年特别火的网络框架.okhttp我想大部分Android开发者都不陌生,因为它的到来.是我们 ...

  2. 开源框架相关面试问题-okhttp网络框架面试问题详解

    OkHttp使用简介: ①.准备OkHttpClient对象: 一般app中使用网络可以全部用它,可以将它弄为全局变量,这样就可以共用它的缓存和线程池了. ②.准备Request对象: 很显然它是采用 ...

  3. 「2020 新手必备 」极速入门 Retrofit + OkHttp 网络框架到实战,这一篇就够了!

    老生常谈 什么是 Retrofit ? Retrofit 早已不是什么新技术了,想必看到这篇博客的大家都早已熟知,这里就不啰嗦了,简单介绍下: Retrofit 是一个针对 Java 和 Androi ...

  4. AFNetworking二次封装的那些事

    AFNetworking可是iOS网络开发的神器,大大简便了操作.不过网络可是重中之重,不能只会用AFNetworking.我觉得网络开发首先要懂基本的理论,例如tcp/ip,http协议,之后要了解 ...

  5. OkGo3.0 --真实项目使用和二次封装(转)

    转载:https://blog.csdn.net/jiushiwo12340/article/details/79011480  11.OkGo3.0真实项目使用和二次封装: ====  11.OkG ...

  6. OkHttp框架从入门到放弃,解析图片使用Picasso裁剪,二次封装OkHttpUtils,Post提交表单数据

    OkHttp框架从入门到放弃,解析图片使用Picasso裁剪,二次封装OkHttpUtils,Post提交表单数据 我们这片博文就来聊聊这个反响很不错的OkHttp了,标题是我恶搞的,本篇将着重详细的 ...

  7. 上门洗车APP --- Androidclient开发 之 网络框架封装介绍(二)

    上门洗车APP --- Androidclient开发 之 网络框架封装介绍(二) 前几篇博文中给大家介绍了一下APP中的基本业务及开发本项目使用的网络架构: 上门洗车APP --- Androidc ...

  8. android基于开源网络框架asychhttpclient,二次封装为通用网络请求组件

    网络请求是全部App都不可缺少的功能,假设每次开发都重写一次网络请求或者将曾经的代码拷贝到新的App中,不是非常合理,出于此目的,我希望将整个网络请求框架独立出来,与业务逻辑分隔开,这样就能够避免每次 ...

  9. 网络框架OKHTTP使用场景全解析

    [本文版权归微信公众号"代码艺术"(ID:onblog)所有,若是转载请务必保留本段原创声明,违者必究.若是文章有不足之处,欢迎关注微信公众号私信与我进行交流!] 一.引言 说句实 ...

随机推荐

  1. vue.js定义一个一级的路由 ----由浅入深

    #### 定义一个路由- 实例化一个路由并设置路由映射表 - 实例化里面第一个参数 routes 路由映射表 - routes 里面参数 - path 路由的路径 - component 路由对应的组 ...

  2. cogs——908. 校园网

    908. 校园网 ★★   输入文件:schlnet.in   输出文件:schlnet.out   简单对比 时间限制:1 s   内存限制:128 MB USACO/schlnet(译 by Fe ...

  3. 25、Java并发性和多线程-阻塞队列

    以下内容转自http://ifeve.com/blocking-queues/: 阻塞队列与普通队列的区别在于,当队列是空的时,从队列中获取元素的操作将会被阻塞,或者当队列是满时,往队列里添加元素的操 ...

  4. docker: useful commands

    docker build -t stock_data_repo_instance24 . docker run -v /opt/log:/opt/log -d -it stock_data_repo_ ...

  5. Open Flash Chart图表的JSON格式基本属性详解

    http://blog.csdn.net/wangwenhui11/article/details/4283571 数据文件必须是JSON格式.JSON对象的基本格式: {} 把所有对象都编写在{}里 ...

  6. 2015 Multi-University Training Contest 2 1004 Delicious Apples(DP)

    pid=5303">题目链接 题意:长度为l 的环,有n棵果树,背包容量为k,告诉你k棵苹果树的id.以及每棵树上结的果子数.背包一旦装满要返回起点(id==0) 清空,问你至少走多少 ...

  7. Batch 拷贝远程机器文件到本机指定目录下

    net use * /del /yesNET USE Y: \\远程机IP\d$  登录密码 /user:domain\登录用户 set sourcePath="Y:\DOAutomatio ...

  8. ALSA声卡驱动中的DAPM详解之一:kcontrol

    DAPM是Dynamic Audio Power Management的缩写,直译过来就是动态音频电源管理的意思,DAPM是为了使基于linux的移动设备上的音频子系统,在任何时候都工作在最小功耗状态 ...

  9. Linux ALSA声卡驱动之七:ASoC架构中的Codec

    1.  Codec简介(ad/da) 在移动设备中,Codec的作用可以归结为4种,分别是: 对PCM等信号进行D/A转换,把数字的音频信号转换为模拟信号 对Mic.Linein或者其他输入源的模拟信 ...

  10. 如何的退出无响应的 SSH 连接

    ~. 具体操作是Shift+-,然后松开按.. tips如果无效,可以先按下Enter,然后进行上面的操作.