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. linux下git+github个人使用记录

    Linux: 安装git的命令: sudo apt install git 查看版本确认安装成功: git --version 生成密钥: ssh-keygen -t rsa -C "you ...

  2. Linux下汇编语言学习笔记72 ---

    这是17年暑假学习Linux汇编语言的笔记记录,参考书目为清华大学出版社 Jeff Duntemann著 梁晓辉译<汇编语言基于Linux环境>的书,喜欢看原版书的同学可以看<Ass ...

  3. HDU——1133 Buy the Ticket

    Buy the Ticket Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) T ...

  4. C++学习之动态数组类的封装

    动态数组(Dynamic Array)是指动态分配的.可以根据需求动态增长占用内存的数组.为了实现一个动态数组类的封装,我们需要考虑几个问题:new/delete的使用.内存分配策略.类的四大函数(构 ...

  5. 转 java面试题

    ● 简述synchronized?Object:Monitor机制: ● 简述happen-before规则 : ● JUC和Object : Monitor机制区别是什么 : 简述AQS原理 : ● ...

  6. scikit-learn:4.7. Pairwise metrics, Affinities and Kernels

    參考:http://scikit-learn.org/stable/modules/metrics.html The sklearn.metrics.pairwise submodule implem ...

  7. Android MaoZhuaWeiBo 好友动态信息列表数据抓取 -3

    前面2篇把大致的开发说的几乎相同了,接下来说说粉丝动态消息列表或时间线数据的抓取与解析显示,我将他所有写在了一个 类里.并以封装类对象的形式存储数据.以下看看基本的服务代码: 粉丝动态消息列表数据抓取 ...

  8. 逆向工程之App脱壳

    http://www.cnblogs.com/ludashi/p/5725743.html iOS逆向工程之App脱壳 本篇博客以微信为例,给微信脱壳."砸壳"在iOS逆向工程中是 ...

  9. 二分查找法(binary search)

    二分查找法:一种在有序列表中查找某个值的算法,它每次都将待查找的空间分为两半,在其中一般继续查找. 使用二分查找的前提是:已经排序好的列表.否则,sum对其查找的结果不做保证. 代码实现: // 使用 ...

  10. windows下写的脚本,在linux下执行失败

    Windows中的换行符为CRLF, 即正则表达式的rn(ASCII码为13和10), 而Unix(或Linux)换行符为LF, 即正则表达式的n. 在Windows和Linux下协同工作的时候, 往 ...