一、简介

Android系统提供了两种HTTP通信类:HttpURLConnection和HttpClient,前者对比后者十分难用。
网络请求进化:HttpURLConnection --- Apache HttpClient --- Volley ---OKHttp
volley是google官方提供的网络请求框架,它依赖于httpclient,而在Android6.0的sdk中去掉了httpclient,因而okhttp更受欢迎。
从Android4.4开始HttpURLConnection的底层实现采用的是okHttp 

二、特点

  • 一般的get请求
  • 一般的post请求
  • 基于Http的文件上传
  • 文件下载
  • 加载图片
  • 支持请求回调,直接返回对象、对象集合
  • 支持session的保持
  • 支持自签名网站https的访问,提供方法设置下证书就行
  • 支持取消某个请求

三、OKHTTP的基本使用

1 引入依赖
compile 'com.squareup.okhttp3:okhttp:3.7.0'
2 使用前,认识一下核心类
  • OKHttpClient 客户端对象
  • Request是OKHttp中的请求,post请求中需要包含RequestBody
  • Build是辅助类,用于生产对象
  • Response是OKHttp中的响应,响应中可以得到返回是否成功,返回数据
  • RequestBody请求数据,在Post请求中用到
  • client.newCall(request).execute()同步请求
  • client.newCall(request).enqueue(CallBack callback)异步请求,但是CallBack里面的代码实在子线程执行的,因此不能更新UI。需要配合Handle使用,更新UI。
3 基本使用步骤

3.1 创建OKHttpClient对象,官方推荐单例。

3.2 创建Request对象,这个对象是请求对象,需要指定URL。

如果是post请求,需要通过FormEncodingBuilder创建RequestBody对象,指定post传进去的参数。get不需要。

3.3 调用okhttpClient的newCall(request).enqueue(CallBack callback)或者execute方法。在callback的回调onResonse函数里做你需要做的事情。onResponse在子线程执行,如需更新UI,配合handler使用。

enqueue是异步请求,有回调接口,execute是同步请求,没有回调接口。一般使用异步请求。

四、基础封装

public class OKHttpHelper {

    private static OKHttpHelper mHelper = null;
private static OkHttpClient mClient = null;
private Handler mHandler; /**
* 私有构造函数
*/
private OKHttpHelper() {
// 没有响应时使用超时结束call。没有响应的原因可能是客户点链接问题、服务器可用性问题或者这之间的其他东西。
mClient = new OkHttpClient().newBuilder()
.connectTimeout(10, TimeUnit.SECONDS)
.writeTimeout(10, TimeUnit.SECONDS)
.readTimeout(10, TimeUnit.SECONDS).build();
mHandler = new Handler(Looper.getMainLooper());
} /**
* 单例模式 获取实例
*
* @return
*/
public static OKHttpHelper getInstance() {
if (mHelper == null) {
synchronized (OKHttpHelper.class) {
if (mHelper == null) {
mHelper = new OKHttpHelper();
}
}
}
return mHelper;
} /**
* 对外公开的post方法
* <p>
* post请求需要通过FormEncodingBuilder创建RequestBody对象,指定post传进去的参数
*
* @param url 请求地址
* @param params 请求参数
* @param callBack 回调接口
* @param tag 请求标签
*/
public void post(String url, Map<String, String> params, BaseCallBack callBack, int tag) throws Exception {
Request req = buildRequest(url, params, tag);
request(req, callBack);
} /**
* 对外公开的get方法
*
* @param url
* @param callBack
*/
public void get(String url, BaseCallBack callBack, int tag) throws Exception {
Request req = buildRequest(url, null, tag);
request(req, callBack);
} /**
* 构建request请求对象
*
* @param url
* @param params
* @return
*/
private Request buildRequest(String url, Map<String, String> params, int tag) {
Request.Builder builder = new Request.Builder();
builder.url(url);//请求地址
if (null == params) {
builder.get();//get请求
} else {
builder.post(buildRequestBody(params));//post请求 添加参数
}
builder.tag(tag);//tag 区分不同的请求
return builder.build();
} /**
* 通过Map的键值对 构建post请求的RequestBody对象
*
* @param params
* @return
*/
private RequestBody buildRequestBody(Map<String, String> params) {
FormBody.Builder formEncodingBuilder = new FormBody.Builder();
if (params != null) {
for (Map.Entry<String, String> entity : params.entrySet()) {
formEncodingBuilder.add(entity.getKey(), entity.getValue());
}
}
return formEncodingBuilder.build();
} /**
* 接口请求 post,get请求通用
*
* @param request 请求对象
* @param callBack 回调接口
*/
private void request(final Request request, final BaseCallBack callBack) throws Exception {
callBack.onRequestBefore();
mClient.newCall(request).enqueue(new Callback() {//异步访问网络
@Override
public void onFailure(Call call, final IOException exception) {
if (!call.isCanceled()) {//请求未取消
mHandler.post(new Runnable() {
@Override
public void run() {
//请求失败 网络异常
callBack.onFailure(exception);
}
});
}
} @Override
public void onResponse(Call call, final Response response) throws IOException {
if (!call.isCanceled()) {//链接超时 或者其他原因 导致请求取消
if (response.isSuccessful()) {
String res = response.body().string();
mHandler.post(new Runnable() {
@Override
public void run() {
callBack.onSuccess(response, (Integer) response.request().tag());
}
});
} else {
mHandler.post(new Runnable() {
@Override
public void run() {
//请求成功 获取数据异常
callBack.onError(response, response.code(), null);
}
});
}
}
}
});
}
}

回调接口

public abstract class BaseCallBack<T> {

    /**
* 请求之前调用
*/
public abstract void onRequestBefore(); /**
* 请求失败(网络问题)
* @param e
*/
public abstract void onFailure(Exception e); /**
* 请求成功
* @param response
* @param tag
*/
public abstract void onSuccess(Response response, int tag); /**
* 请求成功但是有错误(接口错误,数据解析错误等)
* @param response
* @param errorCode
* @param e
*/
public abstract void onError(Response response,int errorCode,Exception e); }

补充:

OKHttp 源码:http://blog.csdn.net/u012124438/article/details/54236967

AsyncHttpClient : 基于HTTPClient的异步请求框架 http://www.it165.net/pro/html/201406/16421.html

xutils : http://doc.okbase.net/u010870518/archive/125128.html

最新文章

  1. 过段时间逐步使用HTML5新增的web worker等内容
  2. jQuery jsonp无法捕获404、500状态错误
  3. viewpaper
  4. paip.hql的调试故障排查流程总结
  5. Linux上使用Azure CLI来管理Azure
  6. SQL Server 改变数据库的名字
  7. Memcached&#183;Redis缓存的基本操作
  8. tomcat中Servlet的工作机制
  9. 给hive的metastore做JVM优化
  10. 设置 IntelliJ IDEA 智能提醒时忽略大小写
  11. error MSB3073: 命令“copy /y
  12. png转tif
  13. bzoj1663: [Usaco2006 Open]赶集
  14. 返回Json格式结果
  15. 前端实现实时通讯-----ajax长连接
  16. Axure-如何设置圆形组件
  17. linux 信号与多线程
  18. [Openwrt 项目开发笔记]:MySQL配置(六)
  19. Codeforces Round #476 (Div. 2) [Thanks, Telegram!] C
  20. 32_java之TCP和UDP

热门文章

  1. [luogu5339] [TJOI2019]唱、跳、rap和篮球(容斥原理+组合数学)(不用NTT)
  2. LINQ 推迟查询的执行
  3. “百度杯”CTF比赛 十一月场--CrackMe01
  4. Mongo--04 Mongo分片集群
  5. Python学习笔记-列表的增删改查
  6. MacOS Mojave 安装sshpass
  7. 北京师范大学第十五届ACM决赛-重现赛D Disdain Chain (规律+组合数学)
  8. 10年前文章_eclipse下perl环境搭建
  9. 对table最后一行显示与隐藏
  10. pymysql基本操作